【题目链接】 http://acm.hdu.edu.cn/showproblem.php?pid=6035

【题目大意】

  给出一颗树,一条路径的价值为其上点权的种类数,求路径总价值

【题解】

  我们计算单个颜色的贡献,那么就是经过该颜色至少一次的路径数量,
  我们统计的时候在每个点记录以其为开始的路径的答案和,
  统计的时候计算了点自身,同时有重复计算的部分,最后减去n除以2即可
  那么我们只要在每种颜色的虚树上统计即可。
  对于子树的贡献需要区间修改,我们在dfs序的差分数组上更改,最后求前缀和即可。  

【代码】

  1. #include <cstdio>
  2. #include <algorithm>
  3. #include <cstring>
  4. #include <vector>
  5. #include <list>
  6. using namespace std;
  7. const int M=200010,N=(M<<1)+10;
  8. int n,x,y,pre[N],st[N],en[N],c[N];
  9. vector<int> u[N],v[N];
  10. long long ans[N];
  11. list<int> l[N];
  12. int dfn;
  13. void dfs(int x,int fx){
  14. int cx=c[x];
  15. if(l[cx].empty())u[M+cx].push_back(x);
  16. else u[l[cx].back()].push_back(x);
  17. pre[x]=fx;
  18. l[cx].push_back(x);
  19. st[x]=dfn++;
  20. for(int i=0;i<v[x].size();i++){
  21. int y=v[x][i];
  22. if(y==fx)continue;
  23. dfs(y,x);
  24. }l[cx].pop_back();
  25. en[x]=dfn;
  26. }
  27. bool isson(int x,int y){return st[y]<=st[x]&&st[x]<en[y];}
  28. void Dfs(int x,int d){
  29. int pos=0;
  30. if(x<=M){
  31. ans[st[x]]+=n-d;
  32. ans[st[x]+1]-=n-d;
  33. }
  34. for(int i=0;i<v[x].size();i++){
  35. int y=v[x][i];
  36. if(y==pre[x])continue;
  37. int p=pos,size=en[y]-st[y];
  38. while(p<u[x].size()&&isson(u[x][p],y)){
  39. size-=en[u[x][p]]-st[u[x][p]];
  40. p++;
  41. }ans[st[y]]+=n-size-d;
  42. ans[en[y]]-=n-size-d;
  43. for(int j=pos;j<p;j++)Dfs(u[x][j],n-size);
  44. pos=p;
  45. }
  46. }
  47. int Cas=1;
  48. int main(){
  49. while(~scanf("%d",&n)){
  50. for(int i=1;i<=n;i++)scanf("%d",&c[i]);
  51. memset(ans,0,sizeof(ans));
  52. for(int i=0;i<N;i++)v[i].clear(),u[i].clear(),l[i].clear();
  53. dfn=1;
  54. for(int i=1;i<n;i++){
  55. scanf("%d%d",&x,&y);
  56. v[x].push_back(y);
  57. v[y].push_back(x);
  58. }dfs(1,1);
  59. for(int i=1;i<=M;i++){v[i+M].push_back(1);Dfs(i+M,0);}
  60. for(int i=1;i<=n;i++)ans[i]+=ans[i-1];
  61. long long Ans=0;
  62. for(int i=1;i<=n;i++)Ans+=ans[st[i]];
  63. printf("Case #%d: %lld\n",Cas++,(Ans-n)/2);
  64. }return 0;
  65. }

HDU 6035 Colorful Tree (树形DP)的更多相关文章

  1. HDU 6035 - Colorful Tree | 2017 Multi-University Training Contest 1

    /* HDU 6035 - Colorful Tree [ DFS,分块 ] 题意: n个节点的树,每个节点有一种颜色(1~n),一条路径的权值是这条路上不同的颜色的数量,问所有路径(n*(n-1)/ ...

  2. 2017 Multi-University Training Contest - Team 1 1003&&HDU 6035 Colorful Tree【树形dp】

    Colorful Tree Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others)T ...

  3. hdu 6035:Colorful Tree (2017 多校第一场 1003) 【树形dp】

    题目链接 单独考虑每一种颜色,答案就是对于每种颜色至少经过一次这种的路径条数之和.反过来思考只需要求有多少条路径没有经过这种颜色即可. 具体实现过程比较复杂,很神奇的一个树形dp,下面给出一个含较详细 ...

  4. hdu6035 Colorful Tree 树形dp 给定一棵树,每个节点有一个颜色值。定义每条路径的值为经过的节点的不同颜色数。求所有路径的值和。

    /** 题目:hdu6035 Colorful Tree 链接:http://acm.hdu.edu.cn/showproblem.php?pid=6035 题意:给定一棵树,每个节点有一个颜色值.定 ...

  5. 2017ACM暑期多校联合训练 - Team 1 1003 HDU 6035 Colorful Tree (dfs)

    题目链接 Problem Description There is a tree with n nodes, each of which has a type of color represented ...

  6. HDU-6035 Colorful Tree(树形DP) 2017多校第一场

    题意:给出一棵树,树上的每个节点都有一个颜色,定义一种值为两点之间路径中不同颜色的个数,然后一棵树有n*(n-1)/2条 路径,求所有的路径的值加起来是多少. 思路:比赛的时候感觉是树形DP,但是脑袋 ...

  7. HDU 6035 Colorful Tree(补集思想+树形DP)

    [题目链接] http://acm.hdu.edu.cn/showproblem.php?pid=6035 [题目大意] 给出一颗树,一条路径的价值为其上点权的种类数,求路径总价值 [题解] 单独考虑 ...

  8. HDU 6035 Colorful Tree(dfs)

    题意:一棵有n个点的树,树上每个点都有颜色c[i],定义每条路径的值为这条路径上经过的不同颜色数量和.求所有路径的值的和. 可以把问题转化为对每种颜色有多少条不同的路径至少经过这种颜色的点,然后加和. ...

  9. hdu 6035 Colorful Tree(虚树)

    考虑到树上操作:首先题目要我们求每条路径上出现不同颜色的数量,并把所有加起来得到答案:我们知道俩俩点之间会形成一条路径,所以我们可以知道每个样例的总的路径的数目为:n*(n-1)/2: 这样单单的求, ...

随机推荐

  1. Vuejs - 组件式开发

    初识组件 组件(Component)绝对是 Vue 最强大的功能之一.它可以扩展HTML元素,封装可复用代码.从较高层面讲,可以理解组件为自定义的HTML元素,Vue 的编译器为它添加了特殊强大的功能 ...

  2. sublime3 快捷键大全

    Ctrl+Shift+P:打开命令面板Ctrl+P:搜索项目中的文件Ctrl+G:跳转到第几行Ctrl+W:关闭当前打开文件Ctrl+Shift+W:关闭所有打开文件Ctrl+Shift+V:粘贴并格 ...

  3. js_微信分享,监听点击分享,分享成功,取消分享,分享失败回调

    2017-06-13 见代码: function weixinShare() { var links = links = "www.youku.com"; common.get_o ...

  4. Python 关于时间和日期函数使用 -- (转)

    python中关于时间和日期函数有time和datatime   1.获取当前时间的两种方法: import datetime,time now = time.strftime("%Y-%m ...

  5. End to End Sequence Labeling via Bidirectional LSTM-CNNs-CRF论文小结

    本篇论文是卡内基梅隆大学语言技术研究所2016年  arXiv:1603.01354v5 [cs.LG] 29 May 2016 今天先理解一下这个是什么意思:        找到的相关理解:arXi ...

  6. Windows下基于python3使用word2vec训练中文维基百科语料(一)

    在进行自然语言处理之前,首先需要一个语料,这里选择维基百科中文语料,由于维基百科是 .xml.bz2文件,所以要将其转换成.txt文件,下面就是相关步骤: 步骤一:下载维基百科中文语料 https:/ ...

  7. 关于RecylerView:1.在ScrollView的RecylerView滑动事件的处理。2.item之间的距离 小数取整

    1.在ScrollView的RecylerView滑动事件的处理. 在布局文件中在RecylerView外包裹一层相对布局 2.RecylerView item之间的距离 (1)编写SpaceItem ...

  8. 任务调度框架kunka

    kunka kunka是一个任务调度框架.用户只需要在Task接口中实现自己要执行的功能,并且选择合适的执行器,放入TaskManager中,就可以了完成整个任务了. 实现细节 整个任务信息存放在内存 ...

  9. vue基本介绍

    https://cn.vuejs.org/v2/guide/ Vue (读音 /vjuː/,类似于 view) 是一套用于构建用户界面的渐进式框架.与其它大型框架不同的是,Vue 被设计为可以自底向上 ...

  10. Laravel 调试器 Debugbar 和数据库导出利器 DbExporter 扩展安装及注意事项

    一.Debugbar安装 参考:Laravel 调试利器 —— Laravel Debugbar 扩展包安装及使用教程 的“2.安装”部分 二.DbExporter安装 参考:Laravel 扩展推荐 ...