题目链接


题解

题意

一棵树上有\(m\)条路径,可以将其中一条边的权值改为0,问最长的路径最短是多少

分析

  • 最短的路径最长自然想到二分最长路径,设其为\(dis\)
  • 关键在于如何check
  • check的关键又是将哪条边改为0
  • 贪心,如果所有超过\(dis\)的路径能在一条边上重合,则将那条边改为0,之后再判断改为0后是否最大的路径小于\(dis\);若无法将所有超过\(dis\)的边重合在一条边上,直接return false;

做法

  • 算两个点之间的路径长用dfs + LCA来实现
  • 判断路径之间的重合用树上差分来实现
  • 这里用的是倍增

注意事项

无向边要把数组开两倍!!!

代码

  1. #include <iostream>
  2. #include <cstdlib>
  3. #include <cstdio>
  4. #include <cstring>
  5. using namespace std;
  6. const int MAXN = 500500;
  7. int n, m;
  8. int logn;
  9. int u[MAXN], v[MAXN], lca[MAXN];
  10. int vis[MAXN], dep[MAXN], fa[MAXN][25];
  11. int dfn[MAXN], Index, Edge_cnt;
  12. int Max_dis = -1, treeC[MAXN], predis[MAXN], dissum[MAXN], distence[MAXN];
  13. int ecnt;
  14. struct node
  15. {
  16. int v;
  17. int w;
  18. node *next;
  19. }pool[MAXN << 1], *head[MAXN << 1];
  20. void addedge(int u, int v, int w)
  21. {
  22. node *p = &pool[++ecnt], *q = &pool[++ecnt];
  23. p->v = v, p->w = w, p->next = head[u], head[u] = p;
  24. q->v = u, q->w = w, q->next = head[v], head[v] = q;
  25. }
  26. void dfs(int u)
  27. {
  28. int v;
  29. dfn[++Index] = u;
  30. vis[u] = 1;
  31. for(node *p = head[u]; p; p = p->next)
  32. if(!vis[v = p->v])
  33. {
  34. dep[v] = dep[u] + 1;
  35. dissum[v] = dissum[u] + p->w;
  36. fa[v][0] = u;
  37. predis[v] = p->w;
  38. dfs(v);
  39. }
  40. }
  41. int LCA(int u, int v)
  42. {
  43. if(dep[u] < dep[v]) swap(u, v);
  44. for(int i = 20; i >= 0; i--)
  45. if(fa[u][i] != 0 && dep[fa[u][i]] >= dep[v])
  46. u = fa[u][i];
  47. if(u == v) return u;
  48. for(int i = 20; i >= 0; i--)
  49. if(fa[u][i] != fa[v][i])
  50. u = fa[u][i], v = fa[v][i];
  51. return fa[u][0];
  52. }
  53. bool check(int dis)
  54. {
  55. memset(treeC, 0, sizeof(treeC));
  56. Edge_cnt = 0;
  57. for(int i = 1; i <= m; i++)
  58. if(distence[i] > dis)
  59. {
  60. ++Edge_cnt;
  61. treeC[u[i]]++;
  62. treeC[v[i]]++;
  63. treeC[lca[i]] -= 2;
  64. }
  65. for(int i = n; i >= 1; i--)
  66. {
  67. int t = dfn[i];
  68. treeC[fa[t][0]] += treeC[t];
  69. if(dis >= Max_dis - predis[t] && treeC[t] == Edge_cnt)
  70. return true;
  71. }
  72. return false;
  73. }
  74. int main()
  75. {
  76. scanf("%d%d", &n, &m);
  77. for(int i = 1; i <= n - 1; i++)
  78. {
  79. int u, v, w;
  80. scanf("%d%d%d", &u, &v, &w);
  81. addedge(u, v, w);
  82. }
  83. dep[1] = 1, dep[0] = -1;
  84. dfs(1);
  85. for(int j = 1; j <= 20; j++)
  86. for(int i = 1; i <= n; i++)
  87. fa[i][j] = fa[fa[i][j - 1]][j - 1];
  88. for(int i = 1; i <= m; i++)
  89. {
  90. scanf("%d%d", &u[i], &v[i]);
  91. lca[i] = LCA(u[i], v[i]);
  92. distence[i] = dissum[u[i]] + dissum[v[i]] - dissum[lca[i]] * 2;
  93. Max_dis = max(Max_dis, distence[i]);
  94. }
  95. int ans = -1;
  96. int l = 0, r = Max_dis;
  97. while(l <= r)
  98. {
  99. int mid = (l + r) / 2;
  100. if(check(mid)) r = mid - 1, ans = mid;
  101. else l = mid + 1;
  102. }
  103. printf("%d\n", ans);
  104. return 0;
  105. }

题解 【luogu P2680 NOIp提高组2015 运输计划】的更多相关文章

  1. 题解——洛谷 P2680 NOIP提高组 2015 运输计划

    树上差分加上二分答案 详细题解待填坑 #include <cstdio> #include <algorithm> #include <cstring> using ...

  2. 【NOIP】提高组2015 运输计划

    [题意]n个点的树,m条链,求将一条边的权值置为0使得最大链长最小. [算法]二分+树上差分 [题解] 最大值最小化问题,先考虑二分最大链长. 对所有链长>mid的链整体+1(树上差分). 然后 ...

  3. 树型大融合——NOIP提高组2015 D1T3 【运输计划】

    下午用一个小时看了一下树上差分,打了个差分模板,A了3题,真的爽! 题目描述: 公元2044 年,人类进入了宇宙纪元. L 国有 n 个星球,还有 n-1 条双向航道,每条航道建立在两个星球之间,这 ...

  4. 【NOIP2015提高组】运输计划

    https://daniu.luogu.org/problem/show?pid=2680 使完成所有运输计划的时间最短,也就是使时间最长的运输计划耗时最短.最大值最小问题考虑用二分答案,每次chec ...

  5. 题解 【luogu P1541 NOIp提高组2010 乌龟棋】

    题目链接 题解 题意: 有一些格子,每个格子有一定分数. 给你四种卡片,每次可以使用卡片来前进1或2或3或4个格子并拾取格子上的分数 每张卡片有数量限制.求最大分数. 分析 设\(dp[i]\)为第前 ...

  6. 题解 【luoguP1967 NOIp提高组2013 货车运输】

    题目链接 题解 题意 给你一个无向图,求两个点之间的一条路径,使路径上的最小值最大 算法:Kruskal最大生成树+倍增lca 分析 首先容易知道,答案一定在该图的最大生成树上 之后问题便转换成了树上 ...

  7. 题解【luoguP1525 NOIp提高组2010 关押罪犯】

    题目链接 题解 算法: 一个经典的并查集 但是需要用一点贪心的思想 做法: 先将给的冲突们按冲突值从大到小进行排序(这很显然) 然后一个一个的遍历它们 如果发现其中的一个冲突里的两个人在同一个集合里, ...

  8. 题解【luoguP1351 NOIp提高组2014 联合权值】

    题目链接 题意:给定一个无根树,每个点有一个权值.若两个点 \(i,j\) 之间距离为\(2\),则有联合权值 \(w_i \times w_j\).求所有的联合权值的和与最大值 分析: 暴力求,每个 ...

  9. NOIP2015_提高组Day2_3_运输计划

    这题思路很简单: 先对每个询问求距离,对距离由大到小排序, 二分最小距离,验证是否可行,验证时用差分处理: #include<iostream> #include<cstring&g ...

随机推荐

  1. XX出行项目子系统-统计系统设计(定时器项目设计例子)

    一. 引言 目前开发的XX出行系统,需要开发数据统计功能,鉴于约约出行系统已经在运营,并且有新版本的迭代,方便以后下个版本复用,遂新建一个子系统. 二. 架构设计 三. 具体实现 1.MySql数据库 ...

  2. JSP页面中文乱码问题

    $.get()方法到服务器端中文乱码 在jsp页面使用encodeURI(“中文”),在服务器端进行解码 String name = req.getParameter("name" ...

  3. 安卓客户端浏览器ajax注意

    这两天被一个bug搞疯了,就是公司安卓app上我负责的网页死活不进ajax,一开始我用的是post方式提交的,但是参数那一栏没写,直接把参数写在url上了,后来老大跟我说post不写参数会出问题,后来 ...

  4. 剑指offer-二叉树搜索树与双向链表25

    题目描述 输入一棵二叉搜索树,将该二叉搜索树转换成一个排序的双向链表.要求不能创建任何新的结点,只能调整树中结点指针的指向. class Solution: def Convert(self, pRo ...

  5. day-20 tensorflow持久化之入门学习

    如果不对模型参数进行保存,当训练结束以后,模型也在内存中被释放,下一轮又需要对模型进行重新训练,有没有一种方法,可以利用之前已经训练好的模型参数值,直接进行模型推理或者继续训练?这里需要引入一个数据之 ...

  6. 3.hadoop完全分布式搭建

    3.Hadoop完全分布式搭建 1.完全分布式搭建 配置 #cd /soft/hadoop/etc/ #mv hadoop local #cp -r local full #ln -s full ha ...

  7. 20145214实验五 Java网络编程及安全

    20145214实验五 Java网络编程及安全 实验内容 1.掌握Socket程序的编写: 2.掌握密码技术的使用: 3.设计安全传输系统. 实验步骤 我的结对伙伴是 20145219 宋歌,我负责的 ...

  8. apache 多端口访问 配置

    使用本地ip:端口号,或者修改hosts文件+域名的方法来进行本地多站点web调试.     注意这里是用apache 不是iis 1: 安装好AppServ2.5.9软件,官网是:[url]http ...

  9. Window Classes in Win32

    探索Win32系统之窗口类(Window Classes in Win32) Kyle MarshMicrosoft Developer Network Technology GroupMSDN技术组 ...

  10. Android 多屏幕适配 dp和px的关系 最好用dp

    Android 多屏幕适配 dp和px的关系 一直以来别人经常问我,android的多屏幕适配到底是怎么弄,我也不知道如何讲解清楚,或许自己也是挺迷糊. 以下得出的结论主要是结合官方文档进行分析的ht ...