题目传送门: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的更多相关文章

  1. 小测几种python web server的性能

    http://blog.csdn.net/raptor/article/details/8038476 因为换了nginx就不再使用mod_wsgi来跑web.py应用了,现在用的是gevent-ws ...

  2. [福大软工] Z班 团队作业——随堂小测(同学录) 作业成绩

    团队作业--随堂小测(同学录) 作业链接 http://www.cnblogs.com/easteast/p/7763645.html 作业情况 本次作业从原先预计的3小时,有些组打了鸡血连续肝了4. ...

  3. 福州大学软件工程1816 | W班 第8次作业[团队作业,随堂小测——校友录]

    作业链接 团队作业,随堂小测--校友录 评分细则 本次个人项目分数由两部分组成(博客分满分40分+程序得分满分60分) 博客和程序得分表 评分统计图 千帆竞发图 总结 旅法师:实现了更新,导出,查询, ...

  4. MySQL课堂小测

    目录 一.基本知识与操作方法 二.小测具体内容 (一)向数据库表中添加记录 (二)下载并导入world.sql (三)数据库查询与输出 (四)查询数据库并求某字段和 (五)查询数据库并取最大& ...

  5. 随堂小测app(nabcd)

    N 现在,老师想要组织测验,需要提前印制试卷,费时费力,考勤采取传统的点名的方式,过程繁琐且结果水分大. 而随堂小测app通过在线答题,智能定位可以帮助老师掌握学生对知识的掌握程度,了解学生的到客情况 ...

  6. java实验(三)——课堂小测

    这次的课堂小测是用以前生成的那些四则运算的代码,然后将这些题目写到一个文件中,再通过这个文件读取题目的信息,每读入一个答案的时候,遇到星号的时候,等待用户输入然后判断输入的答案是否正确,然后输出小一道 ...

  7. echarts tooltip提示框 自定义小圆点(颜色、形状和大小等等)

    项目是拿 echarts + 百度地图 来做可视化界面,现在到收尾阶段慢慢优化. 先附代码: formatter: function(params) { var result = '' params. ...

  8. 朱晔和你聊Spring系列S1E11:小测Spring Cloud Kubernetes @ 阿里云K8S

    有关Spring Cloud Kubernates(以下简称SCK)详见https://github.com/spring-cloud/spring-cloud-kubernetes,在本文中我们主要 ...

  9. 随堂小测APP使用体验

    随堂小测APP使用体验 先要去注册账号需要填写用户名.密码.手机号.学号/教师号.学校.专业.即可注册,注册成功后,即可登录APP进,登陆进去以后.会有两个界面,课堂和我的,注册.登录简单,通俗易懂, ...

随机推荐

  1. mac上eclipse用gdb调试(转)

    mac上eclipse用gdb调试 With its new OS release, Apple has discontinued the use of GDB in OS X. Since 2005 ...

  2. C#获得类的方法和方法参数

    Type t = typeof(CommonController); StringBuilder str = new StringBuilder(); MethodInfo[] methors = t ...

  3. python两种生成md5的方法

    一. 使用md5包 import md5 src = 'this is a md5 test.' m1 = md5.new() m1.update(src) print m1.hexdigest() ...

  4. c# base和this关键字总结

    base:用于在派生类中实现对基类公有或者受保护成员的访问,但是只局限在构造函数.实例方法和实例属性访问器中.MSDN中小结的具体功能包括:    (1)调用基类上已被其他方法重写的方法.     ( ...

  5. Mifare系列2-非接触卡标准(转)

    本文转自 文/闫鑫原创转载请注明出处http://blog.csdn.net/yxstars/article/details/380799 根据信号发送和接收方式的不同,ISO/IEC14443-3定 ...

  6. 运行 Spark on YARN

    运行 Spark on YARN Spark 0.6.0 以上的版本添加了在yarn上执行spark application的功能支持,并在之后的版本中持续的 改进.关于本文的内容是翻译官网的内容,大 ...

  7. java面向对象设计原则

    原则1:DRY(Don't repeat yourself) 即不要写重复的代码,而是用"abstraction"类来抽象公有的东西.如果你需要多次用到一个硬编码值,那么可以设为公 ...

  8. gulp 基本使用

    1, gulp 依赖node, 使用gulp 之前,要先安装node.  Node 安装完成后,它自带npm. Npm: node package manager 就是node 包管理器. 用过jav ...

  9. [android]判断位置服务是否打开

    public boolean isLocationEnabled() { int locationMode = 0; String locationProviders; if (Build.VERSI ...

  10. hub config

    @echo off title Selenium_Hub cd /d %~dp0 java -jar selenium-server-standalone-2.48.2.jar -role hub - ...