传送门:http://poj.org/problem?id=1741

题意:

求树上两点间路径长度小于k的点对个数

题解:

参考资料

守望的淀粉质略解:https://www.luogu.org/blog/user9012/dian-fen-zhi-lve-xie

粉红兔大佬的淀粉质:https://www.cnblogs.com/PinkRabbit/p/8593080.html

算法步骤

  1. 计算重心位置
  2. 计算答案
  3. 分治子问题继续求解(1~3)

代码:

  1. #include <set>
  2. #include <map>
  3. #include <cmath>
  4. #include <cstdio>
  5. #include <string>
  6. #include <vector>
  7. #include <cstring>
  8. #include <iostream>
  9. #include <algorithm>
  10. using namespace std;
  11. typedef long long LL;
  12. typedef pair<int, int> pii;
  13. typedef unsigned long long uLL;
  14. #define ls rt<<1
  15. #define rs rt<<1|1
  16. #define lson l,mid,rt<<1
  17. #define rson mid+1,r,rt<<1|1
  18. #define bug printf("*********\n")
  19. #define FIN freopen("input.txt","r",stdin);
  20. #define FON freopen("output.txt","w+",stdout);
  21. #define IO ios::sync_with_stdio(false),cin.tie(0)
  22. #define debug1(x) cout<<"["<<#x<<" "<<(x)<<"]\n"
  23. #define debug2(x,y) cout<<"["<<#x<<" "<<(x)<<" "<<#y<<" "<<(y)<<"]\n"
  24. #define debug3(x,y,z) cout<<"["<<#x<<" "<<(x)<<" "<<#y<<" "<<(y)<<" "<<#z<<" "<<z<<"]\n"
  25. const int maxn = 3e5 + 5;
  26. const int INF = 0x3f3f3f3f;
  27. int n, k, Ans;
  28. struct EDGE {
  29. int v, nxt, w;
  30. } edge[maxn << 1];
  31. int head[maxn], tot;
  32. void add_edge(int u, int v, int w) {
  33. edge[tot].v = v;
  34. edge[tot].w = w;
  35. edge[tot].nxt = head[u];
  36. head[u] = tot++;
  37. }
  38. bool vis[10001];
  39. int Root, Tsiz, siz[10001], wt[10001];
  40. int arr[10001], cnt;
  41. void GetRoot(int u, int f) {
  42. siz[u] = 1; wt[u] = 0;
  43. for(int i = head[u]; i != -1; i = edge[i].nxt) {
  44. int v = edge[i].v;
  45. if(v != f && !vis[v]) {
  46. GetRoot(v, u);
  47. siz[u] += siz[v];
  48. wt[u] = max(wt[u], siz[v]);
  49. }
  50. }
  51. wt[u] = max(wt[u], Tsiz - siz[u]);
  52. if(wt[Root] > wt[u]) Root = u;
  53. }
  54. void Dfs(int u, int D, int f) {
  55. arr[++cnt] = D;
  56. for(int i = head[u]; i != -1; i = edge[i].nxt) {
  57. int v = edge[i].v;
  58. if(v != f && !vis[v]) {
  59. Dfs(v, D + edge[i].w, u);
  60. }
  61. }
  62. }
  63. int calc(int u, int D) {
  64. cnt = 0; Dfs(u, D, 0); int l = 1, r = cnt, sum = 0;
  65. sort(arr + 1, arr + cnt + 1);
  66. for(;; ++l) {
  67. while(r && arr[l] + arr[r] > k) --r;
  68. if(r < l) break;
  69. sum += r - l + 1;
  70. }
  71. return sum;
  72. }
  73. void DFS(int u) {
  74. Ans += calc(u, 0); vis[u] = 1;
  75. for(int i = head[u]; i != -1; i = edge[i].nxt) {
  76. int v = edge[i].v;
  77. if(!vis[v]) {
  78. Ans -= calc(v, edge[i].w);
  79. Root = 0, Tsiz = siz[v], GetRoot(v, 0);
  80. DFS(Root);
  81. }
  82. }
  83. }
  84. int main() {
  85. #ifndef ONLINE_JUDGE
  86. FIN
  87. #endif
  88. while(~scanf("%d%d", &n, &k) && n && k) {
  89. tot = Ans = 0;
  90. memset(vis, 0, sizeof(vis));
  91. memset(head, -1, sizeof(head));
  92. for(int i = 2, u, v, w; i <= n; i++) {
  93. scanf("%d%d%d", &u, &v, &w);
  94. add_edge(u, v, w);
  95. add_edge(v, u, w);
  96. }
  97. wt[0] = INF;
  98. Tsiz = n;
  99. GetRoot(1, 0);
  100. DFS(Root);
  101. printf("%d\n", Ans - n);
  102. }
  103. return 0;
  104. }

POJ1741 点分治模板的更多相关文章

  1. POJ1741 tree (点分治模板)

    题目大意: 给一棵有 n 个顶点的树,每条边都有一个长度(小于 1001 的正整数).定义 dist(u,v)=节点 u 和 v 之间的最小距离.给定一个整数 k,对于每一对 (u,v) 顶点当且仅当 ...

  2. 洛谷P2634 [国家集训队]聪聪可可 点分治模板

    题意 在一棵树上任意选两个点,求它们距离模3为0的概率. 分析 树分治模板 Code #include<bits/stdc++.h> #define fi first #define se ...

  3. (模板)luoguP3806(树上点分治模板题)

    点分治的写法1: 题目链接:https://www.luogu.org/problem/P3806 题意:给出一颗带边权的树,结点数n<=1e4,每条边有权值<=1e4,有m组询问(m&l ...

  4. [POJ1741]Tree(点分治模板)

    传送门 良心解析 其实以前在求某段序列上的区间统计问题时就碰到过类似于这样的思想. 当时的区间统计问题思路大致是这样: 选取一个点作为中间点,从这个点的左边和右边统计出满足条件的点对.然后当前的中间点 ...

  5. POJ1741 Tree 树分治模板

    http://poj.org/problem?id=1741   题意:一棵n个点的树,每条边有距离v,求该树中距离小于等于k的点的对数.   dis[y]表示点y到根x的距离,v代表根到子树根的距离 ...

  6. 点分治模板(洛谷P4178 Tree)(树分治,树的重心,容斥原理)

    推荐YCB的总结 推荐你谷ysn等巨佬的详细题解 大致流程-- dfs求出当前树的重心 对当前树内经过重心的路径统计答案(一条路径由两条由重心到其它点的子路径合并而成) 容斥减去不合法情况(两条子路径 ...

  7. Luogu P3806 点分治模板1

    题意: 给定一棵有n个点的树询问树上距离为k的点对是否存在. 分析: 这个题的询问和点数都不多(但是显然暴力是不太好过的,即使有人暴力过了) 这题应该怎么用点分治呢.显然,一个模板题,我们直接用套路, ...

  8. POJ - 1741 - Tree - 点分治 模板

    POJ-1741 题意: 对于带权的一棵树,求树中距离不超过k的点的对数. 思路: 点分治的裸题. 将这棵树分成很多小的树,分治求解. #include <algorithm> #incl ...

  9. 【Luogu】P3806点分治模板(点分治)

    题目链接 wc听不懂lca讲的高等数学专场(一个字都听不懂),然后就自学了点分治. 点分治就是我先处理完跟根有关的东西,然后把根标记掉,把原树拆成若干个联通块,然后分别对每个联通块(每个小树)搞一模一 ...

随机推荐

  1. Kubernetes1.3新特性:支持GPU

    (一)  背景资料 GPU就是图形处理器,是Graphics Processing Unit的缩写.电脑显示器上显示的图像,在显示在显示器上之前,要经过一些列处理,这个过程有个专有的名词叫" ...

  2. [React Native] 解析JSON文件

    在编写代码时,开发者有时需要存储一些比较多,在应用程序运行时不需要更改的数据.文件大不便于写在代码中,可以把这些数据存储到JSON文件中. 优点非常明显: 1. 数据存放在单独的文件中,代码精简有条理 ...

  3. TZOJ 4359: Partition the beans (二分)

    描述 Given an N x N square grid (2 <= N <= 15) and each grid has some beans in it. You want to w ...

  4. uml图的五种关系 标签: uml 2016-12-18 21:47 221人阅读 评论(25) 收藏

    统一建模语言 Unified Modeling Language (UML)又称统一建模语言或标准建模语言,是始于1997年一个OMG标准,它是一个支持模型化和软件系统开发的图形化语言,为软件开发的所 ...

  5. 三角形数且是完全平方数 2017 ACM-ICPC 亚洲区(乌鲁木齐赛区)网络赛 E.Half-consecutive Numbers

    三角形数:an=n*(n+1)/2; 完全平方数:bn=c^2; 既是三角形数又是完全平方数:An=6*A(n-1)-A(n-2)+2; A[23]={ 0, 1, 8, 49, 288, 1681, ...

  6. Jmeter xpath处理器

  7. 威胁快报|挖矿团伙8220进化,rootkit挖矿趋势兴起

    近日,阿里云安全团队发现8220挖矿团伙为了更持久的驻留主机以获得最大收益,开始使用rootkit技术来进行自我隐藏.这类隐藏技术的使用在watchdogs等挖矿蠕虫使用后开始出现逐渐扩散和进化的趋势 ...

  8. 2018-8-10-WPF-使用-VisualStudio-2017-项目文件

    title author date CreateTime categories WPF 使用 VisualStudio 2017 项目文件 lindexi 2018-08-10 19:16:53 +0 ...

  9. Laravel5使用ElasticSearch

    https://blog.csdn.net/qq_16829085/article/details/80725125 安装elasticsearch和ik插件 (elasticsearch的使用需要配 ...

  10. 数据采集之js埋点

    一.后台nginx环境搭建 web点数据采集后台配置nginx:https://blog.csdn.net/weixin_37490221/article/details/80894827 下载数据源 ...