日常小测:颜色 && Hackerrank Unique_colors
题目传送门:https://www.hackerrank.com/challenges/unique-colors
感谢hzq大神找来的这道题。
考虑点分治(毕竟是路经统计),对于每一个颜色,它的贡献是独立的。我们可以在一次点分治中合在一块处理(为什么时间复杂度是对的呢,因为我们每次改动只会根据当前点的颜色进行变动,而不是所有颜色对着它都来一遍)。每次先对重心单独计算答案贡献,此时也将当前区域的各个答案贡献计算出来,并以此为基础(之后称之为基准贡献,即代码中的tot)。对于每一棵子树,我们先dfs取消掉这一片区域内贡献(为什么要取消,因为这里的贡献要留到它们那一层去解决,不然会重复计算),然后单独对这里再深搜一遍 ,具体内容如下:
出现一种颜色,如果深搜过程中这是第一次出现,我们要把目前的基准贡献扣除掉该颜色的贡献(由于进入这种深搜之前,已经取消掉这一棵子树的贡献,所以实际扣除掉了外面子树的该颜色贡献)再加上外面子树大小,从而做到维护基准贡献,然后直接加给这个点就形成当前贡献了。当然如果不是第一次出现就不用管了,因为深搜来的路上已经处理过了,然后直接加上。
这是点分治做法,时间复杂度(nlogn)。还有线性做法,即用dfs序,然后对于一个区间打上差分,最后扫一遍。
点分治代码:
#include<bits/stdc++.h>
using namespace std;
#define N 100005
#define INF 1e9
#define LL long long
inline int read(){
int x=,f=; char a=getchar();
while(a<'' || a>'') {if(a=='-') f=-; a=getchar();}
while(a>='' && a<='') x=x*+a-'',a=getchar();
return x*f;
}
int n,cnt,sum,size[N],head[N],a[N],s[N],weight,weights,tot;
LL ans[N];
bool vis[N],app[N];
struct edges{
int to,next;
}e[*N];
inline void insert(){
int u=read(),v=read();
e[cnt]=(edges){v,head[u]};head[u]=cnt++;
e[cnt]=(edges){u,head[v]};head[v]=cnt++;
}
void getroot(int x,int fa){
size[x]=; int tmp=;
for(int i=head[x];i>=;i=e[i].next){
if(vis[e[i].to] || fa==e[i].to) continue;
getroot(e[i].to,x); size[x]+=size[e[i].to];
tmp=max(tmp,size[e[i].to]);
}
tmp=max(tmp,sum-size[x]);
if(tmp<weights) weight=x,weights=tmp;
}
void dfs(int x,int fa,int f){
bool ok=;
if(!app[a[x]] && a[x]!=a[weight]) app[a[x]]=ok=,s[a[x]]+=size[x]*f,tot+=size[x]*f;
for(int i=head[x];i>=;i=e[i].next) if(!vis[e[i].to] && fa!=e[i].to) dfs(e[i].to,x,f);
if(ok) app[a[x]]=;
}
void DFS(int x,int fa,int p){
bool ok=;
if(!app[a[x]] && a[x]!=a[weight]) app[a[x]]=ok=,tot+=p-s[a[x]];
ans[x]+=tot;
for(int i=head[x];i>=;i=e[i].next) if(!vis[e[i].to] && fa!=e[i].to) DFS(e[i].to,x,p);
if(ok) app[a[x]]=,tot-=p-s[a[x]];
}
void work(int x){
weights=INF; getroot(x,); getroot(weight,);
x=weight; vis[x]=;
dfs(weight,,); tot+=size[x]; ans[x]+=tot;
for(int i=head[x];i>=;i=e[i].next){
if(vis[e[i].to]) continue;
dfs(e[i].to,x,-); tot-=size[e[i].to];
DFS(e[i].to,x,size[x]-size[e[i].to]);
dfs(e[i].to,x,); tot+=size[e[i].to];
}
dfs(weight,,-); tot=;
for(int i=head[x];i>=;i=e[i].next)
if(!vis[e[i].to]) sum=size[e[i].to],work(e[i].to);
}
int main(){
n=read(); memset(head,-,sizeof(head));
for(int i=;i<=n;i++) a[i]=read();
for(int i=;i<n;i++) insert();
sum=n; work();
for(int i=;i<=n;i++) printf("%lld\n",ans[i]);
return ;
}
日常小测:颜色 && Hackerrank Unique_colors的更多相关文章
- 小测几种python web server的性能
http://blog.csdn.net/raptor/article/details/8038476 因为换了nginx就不再使用mod_wsgi来跑web.py应用了,现在用的是gevent-ws ...
- [福大软工] Z班 团队作业——随堂小测(同学录) 作业成绩
团队作业--随堂小测(同学录) 作业链接 http://www.cnblogs.com/easteast/p/7763645.html 作业情况 本次作业从原先预计的3小时,有些组打了鸡血连续肝了4. ...
- 福州大学软件工程1816 | W班 第8次作业[团队作业,随堂小测——校友录]
作业链接 团队作业,随堂小测--校友录 评分细则 本次个人项目分数由两部分组成(博客分满分40分+程序得分满分60分) 博客和程序得分表 评分统计图 千帆竞发图 总结 旅法师:实现了更新,导出,查询, ...
- MySQL课堂小测
目录 一.基本知识与操作方法 二.小测具体内容 (一)向数据库表中添加记录 (二)下载并导入world.sql (三)数据库查询与输出 (四)查询数据库并求某字段和 (五)查询数据库并取最大& ...
- 随堂小测app(nabcd)
N 现在,老师想要组织测验,需要提前印制试卷,费时费力,考勤采取传统的点名的方式,过程繁琐且结果水分大. 而随堂小测app通过在线答题,智能定位可以帮助老师掌握学生对知识的掌握程度,了解学生的到客情况 ...
- java实验(三)——课堂小测
这次的课堂小测是用以前生成的那些四则运算的代码,然后将这些题目写到一个文件中,再通过这个文件读取题目的信息,每读入一个答案的时候,遇到星号的时候,等待用户输入然后判断输入的答案是否正确,然后输出小一道 ...
- echarts tooltip提示框 自定义小圆点(颜色、形状和大小等等)
项目是拿 echarts + 百度地图 来做可视化界面,现在到收尾阶段慢慢优化. 先附代码: formatter: function(params) { var result = '' params. ...
- 朱晔和你聊Spring系列S1E11:小测Spring Cloud Kubernetes @ 阿里云K8S
有关Spring Cloud Kubernates(以下简称SCK)详见https://github.com/spring-cloud/spring-cloud-kubernetes,在本文中我们主要 ...
- 随堂小测APP使用体验
随堂小测APP使用体验 先要去注册账号需要填写用户名.密码.手机号.学号/教师号.学校.专业.即可注册,注册成功后,即可登录APP进,登陆进去以后.会有两个界面,课堂和我的,注册.登录简单,通俗易懂, ...
随机推荐
- winform常用的属性(listview),常用容器(二者结合)
ListVies控件主要用于展示数据(非常实用和重要的控件) FullRowSelect:设置是否行选择模式.(默认为false) (开启之后一下选中一行数据) ___________________ ...
- -[NSBundle initWithURL:]: nil URL argument'
今天早上同事突然跟我说趣拍的SDK不能用了,一调用就crash,我一听就纳了闷了,原来好好的啊. 然后就开始查呗,马上就要上线了,不搞好,老大会不会杀了我... 搞个全局断点,就停在了一堆我看不懂的界 ...
- UVA 11464 偶数矩阵
https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem& ...
- Python中MySQLdb模块的安装
安装 MySQLdb是Python语言访问mysql数据库的一个模块,如果你不确定自己的Python环境中是否已经安装了这个模块,可以打开Python shell,输入import MySQLdb,如 ...
- Javascript学习笔记:3种检测变量类型的方法
①typeof检测变量类型 console.log(typeof 1);//number console.log(typeof "a");//string console.log( ...
- Linux C++线程池
.为什么需要线程池? 部分应用程序需要执行很多细小的任务,对于每个任务都创建一个线程来完成,任务完成后销毁线程,而这就会产生一个问题:当执行的任务所需要的时间T1小于等于创建线程时间T2和销毁线程时间 ...
- 发布一个开源极致的javascript模板引擎tpl.js
tpl.js(大家直接去https://git.oschina.net/tianqiq/tpl.js这个上面看) 简介 tpl.js是一个比较极致(极小,极快,极简单)的js模板引擎,可以在各种js环 ...
- Sublime Text 配置记录
sublime userSetting sublime theme sublime plug sublime userSetting 对sublime的配置 { "color_scheme& ...
- TC250专场
SRM 623 DIV2 1000pt 题意:给出一个最多50*50的矩阵,每个单元可能为'.'.'P'.'A','.'代表空地,你每次操作可以把一个P或者A拿到空地上,求一个最大的含有相同字符的矩形 ...
- position之fixed固定定位、absolute绝对定位和relative相对定位
什么是层模型? 什么是层布局模型?层布局模型就像是图像软件PhotoShop中非常流行的图层编辑功能一样,每个图层能够精确定位操作,但在网页设计领域,由于网页大小的活动性,层布局没能受到热捧.但是在网 ...