当前位置:   article > 正文

【学习笔记】详解换根法(换根DP)

【学习笔记】详解换根法(换根DP)

一.换根DP的概念

1.换根DP是什么?

换根DP,又叫二次扫描,是树形DP的一种。

2.换根DP能解决什么问题?

换根DP能解决不指定根结点,并且根节点的变化会对一些值产生影响的问题。例如子结点深度和、点权和等。如果要 暴力求解出最优解,则我们可以枚举所有的节点为根,然后分别跑一次搜索,这样的时间复杂度会达到 O(n^{2}),显然不可接受。这时可以考虑使用换根DP解决。

3.换根DP与一般的树形DP相比有什么不同?

其相比于一般的树形DP具有以下特点:

  • 树上的不同点作为根,其解不同
  • 为其求解答案,不能单求某点的信息,需要求解每个节点的信息
  • 无法通过一次搜索完成答案的求解,因为一次搜索只能得到一个节点的答案。

二.换根DP的解法

换根dp通常会与树形dp 结合,我们可以先任定一个根节点root,通过树形dp的思想计算出一个答案。但这样求解出的答案只是以root为根的解,并不能确定是不是最优解。于是再考虑当root的子节点作为根节点时,答案怎样变化。一般可以在 O(1) 的时间复杂度内完成答案的转化。这样整道题的时间复杂度就由 O(n^{2}) 降为了 O(n)

总结一下,换根DP总共是三个步骤:

  1. 指定某个节点为根节点
  2. 第一次搜索完成预处理(如子树大小等),同时得到以上述节点为根的解
  3. 第二次搜索进行换根的动态规划,由已知解的节点推出相连节点的解。

如果到了这里还没有听懂的话,不用慌,可以结合下面的例题慢慢理解。

三.例题精讲

1.STA-Station

question

sol

我们先来画一下样例~(这里先假设以1号节点为根)

在这幅图中,siz[i]表示以i节点为根的子树中节点个数(包括i节点本身),而deep[i]表示i在以1号节点为根时的深度(1号节点深度为1)。

注意:代码中其实不需要开deep数组(第一次dfs时可以直接在dfs设置一个变量dep,然后每次递归时直接将以一号节点为根的解加上dep就行了),但是为了方便解释,我还是画上了。

这样,我们就可以得出来以1为根时深度之和(也就是解)为1+2+3+3+3+4+5+5=26。

然后我们就可以进行换根操作(比如此时以1的子节点4号节点为根)

siz我们依然沿用第一次得到的结果,deep我为了方便演示已经更新了。

从中我们可以发现一些规律。这棵树中从前(以1号节点为根时)属于4的子树的节点(2,3,4,5,6,7,8)的深度都减了1,而其他的节点(这里样例中只有1)的深度都加了1。大家可以自行在纸上多画几个换根后的图,加深理解。

归纳一下我们得到的结论:

如果此时要将根节点换成节点x

那么,以x为根时的解就是:

以x的父节点为根的子树的节点数-以x为根的子树的节点数+(总节点数-以点为根的子树的节点数)

              ↑                                                     ↑                                               ↑

以x为根时的解的基数     这些节点的深度都要-1(在这里省略了乘1)     其他的节点的深度都要+1

假设dh[x]代表以x为根时的解,fa为x的父节点

dh[x] = dh[fa] - siz[x] + (n - siz[x])

总结一下:

  • 我们假设此时将根节点换成节点x,则其子树由两部分构成,第一部分是其原子树,第二部分则是1号节点的其他子树
  • 根从1号节点变为节点x的过程中,我们可以发现第一部分的深度降低了1第二部分的深度则上升了1,而这两部分节点的数量在第一次搜索时就得到了(siz数组)。
  • DP公式:dh[x] = dh[fa] - siz[x] + (n - siz[x])

最后还要注意:第二次dfs后我们还要枚举dh[1~n]取其中的最大值才能得到答案!

code

  1. #include <bits/stdc++.h>
  2. #define int long long
  3. using namespace std;
  4. vector<int> vec[2000001];
  5. int n,u,v,siz[2000001],dh[2000001],ans;
  6. void dfs(int x,int fa,int dep)
  7. {
  8. siz[x] = 1;//因为以x节点为根的子树包括x自身
  9. dh[1] += dep;//求以1节点为根的解
  10. for(int i = 0; i < vec[x].size(); i++)
  11. {
  12. int son = vec[x][i];
  13. if(son != fa)
  14. {
  15. dfs(son,x,dep + 1);
  16. siz[x] += siz[son];//不断累加子树中的节点个数
  17. }
  18. }
  19. }
  20. void dfs_2(int x,int fa)
  21. {
  22. if(x != 1) dh[x] = dh[fa] - siz[x] + (n - siz[x]);//DP公式
  23. for(int i = 0; i < vec[x].size(); i++)
  24. {
  25. int son = vec[x][i];
  26. if(son != fa) dfs_2(son,x);
  27. }
  28. }
  29. signed main()
  30. {
  31. cin>>n;
  32. for(int i = 1; i < n; i++)
  33. {
  34. cin>>u>>v;
  35. vec[u].push_back(v);//建树
  36. vec[v].push_back(u);
  37. }
  38. dfs(1,0,1);
  39. dfs_2(1,0);
  40. for(int i = 1; i <= n; i++) ans = max(ans,dh[i]);//取以各个节点为根的解的最大值
  41. cout<<ans;
  42. return 0;
  43. }

2.[USACO10MAR] Great Cow Gathering G

question

sol

这道题与例题基本是一致的,只是添加了每个节点和每条边的权值,在DP时注意加上与乘上即可,没有什么大的变形。 如果实在不会,可以结合下面的代码理解。

code

  1. #include <bits/stdc++.h>
  2. #define int long long
  3. using namespace std;
  4. struct ff
  5. {
  6. int len,id;
  7. };
  8. vector<ff> vec[100001];
  9. int n,u,v,siz[1000001],dh[1000001],ans,w,c[1000001],s;
  10. void dfs(int x,int fa)
  11. {
  12. siz[x] = c[x];//以x为根的1子树是包括x节点本身的,所以siz[x]要初始化成x节点上的奶牛个数
  13. for(int i = 0; i < vec[x].size(); i++)
  14. {
  15. int son = vec[x][i].id;
  16. if(son != fa)
  17. {
  18. dfs(son,x);
  19. siz[x] += siz[son];//不断累加以子节点为根的子树上的奶牛数
  20. dh[1] += siz[son] * vec[x][i].len;//因为题目中添加了边权和点权,故不能像sta那题一样直接+dep
  21. }
  22. }
  23. }
  24. void dfs_2(int x,int fa)
  25. {
  26. for(int i = 0; i < vec[x].size(); i++)
  27. {
  28. int son = vec[x][i].id;
  29. if(son != fa)
  30. {
  31. dh[son] = dh[x] - siz[son] * vec[x][i].len + (s - siz[son]) * vec[x][i].len;//此处可以自行画图推导一下
  32. dfs_2(son,x);
  33. }
  34. }
  35. }
  36. signed main()
  37. {
  38. cin>>n;
  39. for(int i = 1; i <= n; i++)
  40. {
  41. cin>>c[i];
  42. s += c[i];//s:奶牛的总数
  43. }
  44. for(int i = 1; i < n; i++)
  45. {
  46. cin>>u>>v>>w;
  47. vec[u].push_back({w,v});//邻接表建双向边
  48. vec[v].push_back({w,u});
  49. }
  50. dfs(1,0);
  51. dfs_2(1,0);
  52. ans = 1e18;
  53. for(int i = 1; i <= n; i++) ans = min(ans,dh[i]);//因为要求深度最小,所以是取解的最小值
  54. cout<<ans;
  55. return 0;
  56. }

四.结语

如果您觉得这篇文章不错,就请点赞关注收藏支持一下吖!ヾ(•ω•`)o

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/article/detail/59482
推荐阅读
相关标签
  

闽ICP备14008679号