Problem Description
 
Coco has a tree, whose vertices are conveniently labeled by 1,2,…,n.
There are m chain on the tree, Each chain has a certain weight. Coco would like to pick out some chains any two of which do not share common vertices.
Find out the maximum sum of the weight Coco can pick
 
  挺不错的一道题目,15年多校的第一场的题目,不过我太弱,比赛的时候没能想出来,赛后想了一天才把他过掉。
  刚开始的时候想法是dfs序,然后dp就好,但是好像有点问题,然后就GG了。。。
  然后想到了树形DP,想了很久才会。
  首先对于每个链,找到他两个端点的LCA,然后把这个链加到他们的LCA那里去。
  dp[i][0]表示对于i这个点,不选LCA为i的链则这颗以i为根的树最大能得到多少,dp[i][1]就是在i的链和不选i的链中最大的那一个。
  然后就是树形DP加上状态转移,dp[i][0]的话状态转移很简单,就是i的所有儿子的dp[i][1]的和就好,但是dp[i][1]这里卡了我老长时间,因为要把所有链上的点的所有非链儿子的dp[i][1]加在一起才行,然后后来想到了用减的方法算,对于某个链上的点x,他的非链儿子的dp[x][1]的和就是他的dp[x][0]减去链上儿子的dp[x][1]就好,所以就是先算出链上所有点的dp[i][0]的和,然后减去除了i的儿子之外链上所有点的dp[i][1]的和就好,再加上除了这个链之外的儿子,就能得到结果了。
  然后算和的话用树链剖分就行。
  好像动态树的话算起来更简单,但是我并不会。。。
 
代码如下:
  1. // ━━━━━━神兽出没━━━━━━
  2. // ┏┓ ┏┓
  3. // ┏┛┻━━━━━━━┛┻┓
  4. // ┃ ┃
  5. // ┃ ━ ┃
  6. // ████━████ ┃
  7. // ┃ ┃
  8. // ┃ ┻ ┃
  9. // ┃ ┃
  10. // ┗━┓ ┏━┛
  11. // ┃ ┃
  12. // ┃ ┃
  13. // ┃ ┗━━━┓
  14. // ┃ ┣┓
  15. // ┃ ┏┛
  16. // ┗┓┓┏━━━━━┳┓┏┛
  17. // ┃┫┫ ┃┫┫
  18. // ┗┻┛ ┗┻┛
  19. //
  20. // ━━━━━━感觉萌萌哒━━━━━━
  21.  
  22. // Author : WhyWhy
  23. // Created Time : 2015年07月22日 星期三 13时55分13秒
  24. // File Name : 1006.cpp
  25.  
  26. #include <stdio.h>
  27. #include <string.h>
  28. #include <iostream>
  29. #include <algorithm>
  30. #include <vector>
  31. #include <queue>
  32. #include <set>
  33. #include <map>
  34. #include <string>
  35. #include <math.h>
  36. #include <stdlib.h>
  37. #include <time.h>
  38.  
  39. #pragma comment(linker, "/STACK:1024000000,1024000000")
  40.  
  41. using namespace std;
  42.  
  43. const int MaxN=;
  44.  
  45. struct Edge
  46. {
  47. int to,next;
  48. };
  49.  
  50. struct Lian
  51. {
  52. int u,v;
  53. int cost;
  54. int next;
  55. };
  56.  
  57. int N,M;
  58.  
  59. Edge E[MaxN<<];
  60. int head[MaxN],Ecou;
  61.  
  62. Lian L[MaxN];
  63. int Lhead[MaxN],Lcou;
  64.  
  65. int fa[MaxN],dep[MaxN],son[MaxN],siz[MaxN],top[MaxN],w[MaxN];
  66. int Tcou;
  67.  
  68. int C_max[MaxN],C_wu[MaxN];
  69.  
  70. void init()
  71. {
  72. Ecou=;
  73. Lcou=;
  74. Tcou=;
  75. w[]=;
  76. top[]=;
  77. memset(Lhead,-,sizeof(Lhead));
  78. memset(head,-,sizeof(head));
  79. }
  80.  
  81. void addEdge(int u,int v)
  82. {
  83. E[Ecou].to=v;
  84. E[Ecou].next=head[u];
  85. head[u]=Ecou++;
  86. }
  87.  
  88. int lca(int a,int b)
  89. {
  90. while()
  91. {
  92. if(top[a]==top[b])
  93. return dep[a]<dep[b] ? a : b;
  94. else if(dep[top[a]]>dep[top[b]])
  95. a=fa[top[a]];
  96. else
  97. b=fa[top[b]];
  98. }
  99. }
  100.  
  101. void addLian(int u,int v,int c)
  102. {
  103. int h=lca(u,v);
  104.  
  105. L[Lcou].u=u;
  106. L[Lcou].v=v;
  107. L[Lcou].cost=c;
  108. L[Lcou].next=Lhead[h];
  109. Lhead[h]=Lcou++;
  110. }
  111.  
  112. void dfs1(int u,int pre,int d)
  113. {
  114. int v;
  115.  
  116. dep[u]=d;
  117. fa[u]=pre;
  118. siz[u]=;
  119. son[u]=-;
  120.  
  121. for(int i=head[u];i!=-;i=E[i].next)
  122. if(E[i].to!=pre)
  123. {
  124. v=E[i].to;
  125. dfs1(v,u,d+);
  126. siz[u]+=siz[v];
  127.  
  128. if(son[u]==- || siz[son[u]]<siz[v])
  129. son[u]=v;
  130. }
  131. }
  132.  
  133. void dfs2(int u)
  134. {
  135. if(son[u]==-)
  136. return;
  137.  
  138. top[son[u]]=top[u];
  139. w[son[u]]=++Tcou;
  140.  
  141. dfs2(son[u]);
  142.  
  143. int v;
  144.  
  145. for(int i=head[u];i!=-;i=E[i].next)
  146. if(E[i].to!=son[u] && E[i].to!=fa[u])
  147. {
  148. v=E[i].to;
  149. top[v]=v;
  150. w[v]=++Tcou;
  151. dfs2(v);
  152. }
  153. }
  154.  
  155. void TL_init()
  156. {
  157. dfs1(,-,);
  158. dfs2();
  159. memset(C_max,,sizeof(C_max));
  160. memset(C_wu,,sizeof(C_wu));
  161. }
  162.  
  163. inline int lowbit(int x)
  164. {
  165. return x&(-x);
  166. }
  167.  
  168. int sum(int x,int C[])
  169. {
  170. int ret=;
  171.  
  172. while(x>)
  173. {
  174. ret+=C[x];
  175. x-=lowbit(x);
  176. }
  177.  
  178. return ret;
  179. }
  180.  
  181. void add(int x,int d,int C[])
  182. {
  183. while(x<=N)
  184. {
  185. C[x]+=d;
  186. x+=lowbit(x);
  187. }
  188. }
  189.  
  190. int query(int u,int v,int C[])
  191. {
  192. int f1=top[u],f2=top[v];
  193. int ret=;
  194.  
  195. while(f1!=f2)
  196. {
  197. if(dep[f1]<dep[f2])
  198. {
  199. swap(f1,f2);
  200. swap(u,v);
  201. }
  202.  
  203. ret+=sum(w[u],C)-sum(w[f1]-,C);
  204. u=fa[f1];
  205. f1=top[u];
  206. }
  207.  
  208. if(dep[u]>dep[v])
  209. swap(u,v);
  210.  
  211. ret+=sum(w[v],C)-sum(w[u]-,C);
  212.  
  213. return ret;
  214. }
  215.  
  216. void update(int u,int ut,int C[])
  217. {
  218. add(w[u],ut,C);
  219. }
  220.  
  221. int dp[MaxN][];
  222.  
  223. void dfs(int u)
  224. {
  225. int v;
  226. int ans=;
  227. int maxn=,tsum1,tsum2;
  228.  
  229. for(int i=head[u];i!=-;i=E[i].next)
  230. if(E[i].to!=fa[u])
  231. {
  232. dfs(E[i].to);
  233. ans+=dp[E[i].to][];
  234. }
  235.  
  236. for(int h=Lhead[u];h!=-;h=L[h].next)
  237. {
  238. tsum1=query(L[h].u,L[h].v,C_wu);
  239. tsum2=query(L[h].u,L[h].v,C_max);
  240. maxn=max(maxn,tsum1-tsum2+ans+L[h].cost);
  241. }
  242.  
  243. maxn=max(maxn,ans);
  244. dp[u][]=ans;
  245. dp[u][]=maxn;
  246.  
  247. update(u,ans,C_wu);
  248. update(u,maxn,C_max);
  249. }
  250.  
  251. int main()
  252. {
  253. //freopen("in.txt","r",stdin);
  254. //freopen("out.txt","w",stdout);
  255.  
  256. int T;
  257. int a,b,c;
  258.  
  259. scanf("%d",&T);
  260.  
  261. while(T--)
  262. {
  263. scanf("%d %d",&N,&M);
  264. init();
  265.  
  266. for(int i=;i<N;++i)
  267. {
  268. scanf("%d %d",&a,&b);
  269. addEdge(a,b);
  270. addEdge(b,a);
  271. }
  272.  
  273. TL_init();
  274.  
  275. while(M--)
  276. {
  277. scanf("%d %d %d",&a,&b,&c);
  278. addLian(a,b,c);
  279. }
  280.  
  281. memset(dp,,sizeof(dp));
  282. dfs();
  283.  
  284. printf("%d\n",dp[][]);
  285. }
  286.  
  287. return ;
  288. }

(中等) HDU 5293 Tree chain problem,树链剖分+树形DP。的更多相关文章

  1. HDU 5293 Train chain Problem - 树链剖分(树状数组) + 线段树+ 树型dp

    传送门 题目大意: 一颗n个点的树,给出m条链,第i条链的权值是\(w_i\),可以选择若干条不相交的链,求最大权值和. 题目分析: 树型dp: dp[u][0]表示不经过u节点,其子树的最优值,dp ...

  2. [HDU 5293]Tree chain problem(树形dp+树链剖分)

    [HDU 5293]Tree chain problem(树形dp+树链剖分) 题面 在一棵树中,给出若干条链和链的权值,求选取不相交的链使得权值和最大. 分析 考虑树形dp,dp[x]表示以x为子树 ...

  3. HDU5293 树链剖分+树形DP

    =-=抓住叶节点往上揪 Tree chain problem Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 65536/65536 K ...

  4. HDU 5293 Tree chain problem 树形dp+dfs序+树状数组+LCA

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5293 题意: 给你一些链,每条链都有自己的价值,求不相交不重合的链能够组成的最大价值. 题解: 树形 ...

  5. HDU 5293 Tree chain problem

    树状数组 + dp 设$f_i$表示以$i$为根的子树中的能选取的最大和,$sum_x$表示$\sum_{f_y}$  ($y$是$x$的一个儿子),这样子我们把所有给出的链按照两点的$lca$分组, ...

  6. codeforces 671D Roads in Yusland & hdu 5293 Tree chain problem

    dp dp优化 dfs序 线段树 算是一个套路.可以处理在树上取链的问题.

  7. HDU 5293 Tree chain problem 树形DP

    题意: 给出一棵\(n\)个节点的树和\(m\)条链,每条链有一个权值. 从中选出若干条链,两两不相交,并且使得权值之和最大. 分析: 题解 #include <cstdio> #incl ...

  8. Water Tree CodeForces 343D 树链剖分+线段树

    Water Tree CodeForces 343D 树链剖分+线段树 题意 给定一棵n个n-1条边的树,起初所有节点权值为0. 然后m个操作, 1 x:把x为根的子树的点的权值修改为1: 2 x:把 ...

  9. [POJ3237]Tree解题报告|树链剖分|边剖

    关于边剖 之前做的大多是点剖,其实转换到边剖非常简单. 我的做法是每个点的点权记录其到父亲节点的边的边权. 只要solve的时候不要把最上面的点记录在内就可以了. Tree Description Y ...

随机推荐

  1. 在ASP.Net中"../" "./" "~/"表示的意思

    "../" 相对当前(根据不同的语境,当前是不同的)的上一级目录"./" 就是当前目录 "~/" 当前项目的根路径,只在服务器端有效.

  2. 【同一直线最多点】 poj 1118+2606+2780

    poj 1118 #include<iostream> using namespace std; #define N 700 struct point {int x,y;} pnt[N]; ...

  3. Cormen — The Best Friend Of a Man

    Cormen — The Best Friend Of a Man time limit per test 1 second memory limit per test 256 megabytes i ...

  4. 初识Jmeter(一)

    倒霉熊的推荐: 文本学习网址:http://m.open-open.com/m/doc/category/105 视频学习网址: 软件学习网:http://www.ask3.cn/index.html ...

  5. android下m、mm、mmm编译命令的使用

    android下m.mm.mmm编译命令的使用 通过查看android源码目录下的build/envsetup.sh文件,可知: - m:       Makes from the top of th ...

  6. javascript 总结学习一

    1 , javascript字符集:javascript采用的是Unicode字符集编码.为什么要采用这个编码呢?原因很简单,16位的Unicode编码可以表示地球人的任何书面语言.这是语言 国际化的 ...

  7. 如何在使用eclipse的情况下,清理android项目中的冗余class文件和资源文件以及冗余图片

    在我们迭代项目的过程中,经常会启用某些功能,或者修改某些界面的问题,那么问题来了,这样很容易出现大量的冗余.java文件,冗余资源文件,一些冗余的界面文件等.那么问题既然出现了,那么如何去解决呢,这就 ...

  8. Leetcode389

    Find the Difference Given two strings s and t which consist of only lowercase letters. 给出两个字符串,s和t,都 ...

  9. Nginx代理外网映射

    外网映射内网端口8080, 外网访问使用端口8379: nginx监听8080和80端口 #user nobody; worker_processes ; #error_log logs/error. ...

  10. 业务零影响!如何在Online环境中巧用MySQL传统复制技术【转】

    业务零影响!如何在Online环境中巧用MySQL传统复制技术 这篇文章我并不会介绍如何部署一个MySQL复制环境或keepalived+双主环境,因为此类安装搭建的文章已经很多,大家也很熟悉.在这篇 ...