1405 树的距离之和

题意

给定一棵无根树,假设它有n个节点,节点编号从1到n,求任意两点之间的距离(最短路径)之和。

分析

树形DP。

首先我们让 \(1\) 为根。要开两个数组 \(up \ down\) 分别记录上面点、下面的点到当前点的距离之和。那么对于每个点答案就是 \(up[i] + down[i]\) 。

\(sons[u]\) 数组表示 \(u\) 以及它下面的所有子孙的数量。

显然 \(down[u]\) 是很好求的,当我们计算到某一点 \(u\) 时,当它的以 v 节点为根的子树递归结束后,有 \(down[u] = down[v] + sons[v]\) ,可以把 \(sons[v]\) 当做下面所有点到 \(u\) 这一点有多少条路径,对于 \(u - v\) 这条边,每一条路径都会算一次贡献。

然后在开个 \(DFS\) 去求 \(up[v]\) ,设 \(u\) 为 \(v\) 的父亲节点,有 \(up[v] = up[u] + (n - sons[u]) + (sons[u] - sons[v]) + (down[u] - down[v] - sons[v])\) ,和上面类似 ,第一个括号算的是所有 u 上面的的节点的数量,第二个括号算的是除了 \(v\) 这棵子树,\(u\) 的其它子树的节点数量,意义就和上面的 \(sons[v]\) 一样,最后一个括号算的是 \(u\) 的其它子树上的节点到 \(u\) 的距离之和。

附上一组数据,模拟完就懂了(树形DP真是在树上找规律啊.....)

  1. 7
  2. 1 2
  3. 2 3
  4. 2 4
  5. 4 6
  6. 4 7
  7. 2 5
  8. ----
  9. 13
  10. 8
  11. 13
  12. 9
  13. 13
  14. 14
  15. 14

code

  1. #include<cstdio>
  2. #include<cstring>
  3. #include<algorithm>
  4. #include<cmath>
  5. #include<set>
  6. #include<vector>
  7. #include<iostream>
  8. #include<map>
  9. using namespace std;
  10. typedef long long ll;
  11. const int MAXN = 1e5 + 10;
  12. const int INF = 1e9;
  13. ll up[MAXN], down[MAXN];
  14. int n, sons[MAXN];
  15. int head[MAXN << 1];
  16. struct edge {
  17. int to, next;
  18. }e[MAXN << 1];
  19. int cnt = 0;
  20. void addedge(int u, int v) {
  21. e[cnt].to = v;
  22. e[cnt].next = head[u];
  23. head[u] = cnt++;
  24. }
  25. void dfs1(int fa, int u) {
  26. sons[u] = 1;
  27. for(int i = head[u]; ~i; i = e[i].next) {
  28. int v = e[i].to;
  29. if(v != fa) {
  30. dfs1(u, v);
  31. sons[u] += sons[v];
  32. down[u] += down[v] + sons[v];
  33. }
  34. }
  35. }
  36. void dfs2(int fa, int u) {
  37. for(int i = head[u]; ~i; i = e[i].next) {
  38. int v = e[i].to;
  39. if(v != fa) {
  40. up[v] = up[u] + (n - sons[u]) + (sons[u] - sons[v]) + (down[u] - down[v] - sons[v]);
  41. dfs2(u, v);
  42. }
  43. }
  44. }
  45. int main() {
  46. scanf("%d", &n);
  47. memset(head, -1, sizeof head);
  48. for(int i = 1; i < n; i++) {
  49. int u, v;
  50. scanf("%d%d", &u, &v);
  51. addedge(u, v);
  52. addedge(v, u);
  53. }
  54. dfs1(0, 1);
  55. dfs2(0, 1);
  56. for(int i = 1; i <= n; i++) {
  57. printf("%lld\n", up[i] + down[i]);
  58. }
  59. return 0;
  60. }

51Nod - 1405 树的距离之和(树形DP)的更多相关文章

  1. 51nod 1405 树的距离之和 树形dp

    1405 树的距离之和 基准时间限制:1 秒 空间限制:131072 KB   收藏  关注 给定一棵无根树,假设它有n个节点,节点编号从1到n, 求任意两点之间的距离(最短路径)之和. Input ...

  2. 51Nod 1405 树的距离之和(dp)

    1405 树的距离之和 基准时间限制:1 秒 空间限制:131072 KB 分值: 40 难度:4级算法题  收藏  关注 给定一棵无根树,如果它有n个节点,节点编号从1到n, 求随意两点之间的距离( ...

  3. 51Nod 1405 树的距离之和 (树dp)

    题目链接:http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1405 中文题面不解释了,两次dfs,第一次自下向上,第二次自上 ...

  4. 51nod 1405 树的距离之和(dfs)

    http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1405 题意: 思路: 先求出所有点到根节点的距离,需要维护每棵子树的大小 ...

  5. 51 nod 1405 树的距离之和

    1405 树的距离之和 基准时间限制:1 秒 空间限制:131072 KB 分值: 40 难度:4级算法题   给定一棵无根树,假设它有n个节点,节点编号从1到n, 求任意两点之间的距离(最短路径)之 ...

  6. [51NOD1405] 树的距离之和(树DP)

    题目链接:http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1405 (1)我们给树规定一个根.假设所有节点编号是0-(n-1 ...

  7. BZOJ5123 线段树的匹配(树形dp)

    线段树的任意一棵子树都相当于节点数与该子树相同的线段树.于是假装在树形dp即可,记忆化搜索实现,有效状态数是logn级别的. #include<iostream> #include< ...

  8. 1113: [视频]树形动态规划(TreeDP)8:树(tree)(树形dp状态设计总结)

    根据最近做的几道树形dp题总结一下规律.(从这篇往前到洛谷 P1352 ) 这几道题都是在一颗树上,然后要让整棵树的节点或边 满足一种状态.然后点可以影响到相邻点的这种状态 然后求最小次数 那么要从两 ...

  9. [CEOI2007]树的匹配Treasury(树形DP+高精)

    题意 给一棵树,你可以匹配有边相连的两个点,问你这棵树的最大匹配时多少,并且计算出有多少种最大匹配. N≤1000,其中40%的数据答案不超过 108 题解 显然的树形DP+高精. 这题是作为考试题考 ...

随机推荐

  1. js限制文本框只能输入特定字符

    限制只能输入数字 // ---------------------------------------------------------------------- // <summary> ...

  2. Percona-Tookit工具包之pt-slave-find

      Preface       If we want to check out how many slaves in our replication environment.We can execut ...

  3. CentOS 7.5 部署蓝鲸运维平台

    环境准备 官方建议 准备至少3台 CentOS 7 以上操作系统的机器 最低配置:2核4G 建议配置: 4核12G 以上 部署前关闭待安装主机之间防火墙,保证蓝鲸主机之间通信无碍 部署前关闭SELin ...

  4. 一个初学者的辛酸路程-继续Django

    问题1:HTTP请求过来会先到Django的那个地方? 先到urls.py  ,里面写的是对应关系,1个URL对应1个函数名. 如果发URL请求过来,到达这里,然后帮你去执行指定的函数,函数要做哪些事 ...

  5. hadoop-eclipse环境搭建(二)

    Eclipse插件配置 第一步:把我们的"hadoop-eclipse-plugin-1.0.0.jar"放到Eclipse的目录的"plugins"中,然后重 ...

  6. 201621123034 《Java程序设计》第11周学习总结

    作业11-多线程 1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结多线程相关内容. 2. 书面作业 本次PTA作业题集多线程 1. 源代码阅读:多线程程序BounceThread ...

  7. hdu1877进制转换

    #include <stdio.h> int m; void Ck(int n) { if(n>=m) Ck(n/m); printf("%d",n%m); } ...

  8. js文字效果

    <!DOCTYPE html> <html> <head> <meta charset="utf-8" /> <title&g ...

  9. 根文件系统制作、NFS配置与安装及利用NFS挂载根文件系统

    最近打算从头开始制作根文件系统,下面是开发过程. 一.根文件系统的制作 0.FHS(Filesystem Hierarchy Standard)标准介绍 该标准规定了根目录下各个子目录的名称及其存放的 ...

  10. 【bzoj2212】[Poi2011]Tree Rotations 权值线段树合并

    原文地址:http://www.cnblogs.com/GXZlegend/p/6826614.html 题目描述 Byteasar the gardener is growing a rare tr ...