也许更好的阅读体验

\(\mathcal{Description}\)

原题链接

概括一下题意

给一颗有\(n\)个点带边权的树,有\(m\)个询问,每次询问\(u,v\)两点间的权值和,你可以将树中任意一条边的边权变为\(0\),问经过一次修改,这\(m\)个询问中最大的权值和最小可以是多少

\(\mathcal{Solution}\)

最大最小,显然二分

二分最小是多少,单调性显然

如何\(check\)

设当前二分到了\(x\),这\(m\)个询问中有\(tot\)个权值和大于\(x\)的询问

那么我们要将一条边变为\(0\),这条边肯定得到被这\(tot\)个询问都经过才行,否则就至少有一条权值和大于\(x\)的询问

那么在所有被这\(tot\)个询问经过的边中,去掉权值最大的那条边肯定最优

之后看一看是否满足,只需考虑原最大的询问是否已经小于等于\(x\)

看一条边是否被这\(tot\)个询问经过,可以用树上差分统计

倍增求\(LCA\)会被卡,卡了我一个下午,后来把二分枚举的上下界调了一下就过了

这告诉我们,二分答案时最好不要无脑二分

\(l\)设初值为最大的询问减去最大的边权,是负数就设为\(0\)

\(r\)设初值为最大的询问

\(\mathcal{Code}\)

为方便阅读,代码中有折叠(扒下来到vim里看吧)

\(val[i]\)是\(i\)的父亲与\(i\)相连的边的权值

  1. /*******************************
  2. Author:Morning_Glory
  3. LANG:C++
  4. Created Time:2019年09月09日 星期一 14时53分08秒
  5. *******************************/
  6. #include <cstdio>
  7. #include <fstream>
  8. #include <algorithm>
  9. using namespace std;
  10. const int maxn = 300005;
  11. const int maxm = 600005;
  12. const int limit = 23;
  13. //{{{cin
  14. struct IO{
  15. template<typename T>
  16. IO & operator>>(T&res){
  17. res=0;
  18. bool flag=false;
  19. char ch;
  20. while((ch=getchar())>'9'||ch<'0') flag|=ch=='-';
  21. while(ch>='0'&&ch<='9') res=(res<<1)+(res<<3)+(ch^'0'),ch=getchar();
  22. if (flag) res=~res+1;
  23. return *this;
  24. }
  25. }cin;
  26. //}}}
  27. int n,m,cnt,num,tot,mx,l,r;
  28. int head[maxn],nxt[maxm],to[maxm],w[maxm],dep[maxn],lg[maxn];
  29. int sum[maxn],u[maxn],v[maxn],val[maxn],len[maxn],dis[maxn],lca[maxn];
  30. int fa[maxn][limit+1];
  31. //{{{add
  32. void add (int u,int v,int val)
  33. {
  34. nxt[++cnt]=head[u],head[u]=cnt,to[cnt]=v,w[cnt]=val;
  35. }
  36. //}}}
  37. //{{{Deal
  38. void Deal (int x)
  39. {
  40. for (int i=1;i<=lg[dep[x]];++i) fa[x][i]=fa[fa[x][i-1]][i-1];
  41. for (int e=head[x];e;e=nxt[e]){
  42. if (to[e]==fa[x][0]) continue;
  43. val[to[e]]=w[e];
  44. fa[to[e]][0]=x;
  45. dep[to[e]]=dep[x]+1;
  46. len[to[e]]=len[x]+w[e];
  47. Deal(to[e]);
  48. }
  49. }
  50. //}}}
  51. //{{{LCA
  52. int LCA (int x,int y)
  53. {
  54. if (dep[x]<dep[y]) swap(x,y);
  55. for (int i=lg[dep[x]];~i;--i)
  56. if (dep[fa[x][i]]>=dep[y]) x=fa[x][i];
  57. if (x==y) return y;
  58. for (rint i=lg[dep[x]];~i;--i)
  59. if (fa[x][i]!=fa[y][i]) x=fa[x][i],y=fa[y][i];
  60. return fa[x][0];
  61. }
  62. //}}}
  63. //{{{dfs
  64. void dfs (int x)
  65. {
  66. for (int e=head[x];e;e=nxt[e]){
  67. if (to[e]==fa[x][0]) continue;
  68. dfs(to[e]);
  69. sum[x]+=sum[to[e]];
  70. sum[to[e]]=0;
  71. if (sum[x]==tot) num=max(num,val[x]);
  72. }
  73. }
  74. //}}}
  75. //{{{check
  76. bool check (int x)
  77. {
  78. sum[1]=num=tot=0;
  79. for (int i=1;i<=m;++i)
  80. if (dis[i]>x){
  81. ++tot;
  82. ++sum[u[i]],++sum[v[i]];
  83. sum[lca[i]]-=2;
  84. }
  85. dfs(1);
  86. return mx-num<=x;
  87. }
  88. //}}}
  89. int main()
  90. {
  91. lg[0]=-1,dep[1]=1;
  92. for (int i=1;i<=maxn-5;++i) lg[i]=lg[i>>1]+1;
  93. cin>>n>>m;
  94. for (int i=2;i<=n;++i){
  95. int x,y,wi;
  96. cin>>x>>y>>wi;
  97. add(x,y,wi),add(y,x,wi);
  98. l=max(l,wi);
  99. }
  100. Deal(1);
  101. for (int i=1;i<=m;++i){
  102. cin>>u[i]>>v[i];
  103. lca[i]=LCA(u[i],v[i]);
  104. dis[i]=len[u[i]]+len[v[i]]-2*len[lca[i]];
  105. mx=max(mx,dis[i]);
  106. }
  107. r=mx,l=max(0,r-l);
  108. while (l<r){
  109. int mid=(l+r)/2;
  110. if (check(mid)) r=mid;
  111. else l=mid+1;
  112. }
  113. printf("%d\n",l);
  114. return 0;
  115. }

如有哪里讲得不是很明白或是有错误,欢迎指正

如您喜欢的话不妨点个赞收藏一下吧

运输计划[二分答案 LCA 树上差分]的更多相关文章

  1. 【bzoj4326】[NOIP2015]运输计划 二分答案+LCA

    题目描述 公元 2044 年,人类进入了宇宙纪元.L 国有 n 个星球,还有 n−1 条双向航道,每条航道建立在两个星球之间,这 n−1 条航道连通了 L 国的所有星球.小 P 掌管一家物流公司, 该 ...

  2. [luogu]P2680 运输计划[二分答案][树上差分]

    [luogu]P2680 [NOIP2015]运输计划 题目背景 公元 2044 年,人类进入了宇宙纪元. 题目描述 L 国有 n 个星球,还有 n-1 条双向航道,每条航道建立在两个星球之间,这 n ...

  3. luogu P2680 运输计划 (二分答案+树上差分)

    题目背景 公元 20442044 年,人类进入了宇宙纪元. 题目描述 公元20442044 年,人类进入了宇宙纪元. L 国有 nn 个星球,还有 n-1n−1 条双向航道,每条航道建立在两个星球之间 ...

  4. loj2425 「NOIP2015」运输计划[二分答案+树上差分]

    看到题意最小化最长路径,显然二分答案,枚举链长度不超过$\text{mid}$,然后尝试检验.````` 检验是否存在这样一个边置为0后,全部链长$\le\text{mid}$,其最终目标就是.要让所 ...

  5. 【BZOJ-4326】运输计划 树链剖分 + 树上差分 + 二分

    4326: NOIP2015 运输计划 Time Limit: 30 Sec  Memory Limit: 128 MBSubmit: 703  Solved: 461[Submit][Status] ...

  6. vijos 运输计划 - 二分答案 - 差分 - Tarjan

    Description 公元 2044 年,人类进入了宇宙纪元.L 国有 n 个星球,还有 n−1 条双向航道,每条航道建立在两个星球之间,这 n−1 条航道连通了 L 国的所有星球.小 P 掌管一家 ...

  7. BZOJ 4326: NOIP2015 运输计划(二分,树上差分)

    Time Limit: 30 Sec  Memory Limit: 128 MBSubmit: 1945  Solved: 1243[Submit][Status][Discuss] Descript ...

  8. 洛谷 P2680 运输计划-二分+树上差分(边权覆盖)

    P2680 运输计划 题目背景 公元 20442044 年,人类进入了宇宙纪元. 题目描述 公元20442044 年,人类进入了宇宙纪元. L 国有 nn 个星球,还有 n-1n−1 条双向航道,每条 ...

  9. NOIP2015 运输计划 - 二分 + 树链剖分 / (倍增 + 差分)

    BZOJ CodeVS Uoj 题目大意: 给一个n个点的边带权树,给定m条链,你可以选择树中的任意一条边,将它置为0,使得最长的链长最短. 题目分析: 最小化最大值,二分. 二分最短长度mid,将图 ...

随机推荐

  1. 通过字节码分析this关键字以及异常表的作用

    1.创建MyTest3类 public class MyTest3 { public void test(){ try { InputStream is = new FileInputStream(& ...

  2. python 设计模式之单例模式 Singleton Pattern

    #引入 一个类被设计出来,就意味着它具有某种行为(方法),属性(成员变量).一般情况下,当我们想使用这个类时,会使用new 关键字,这时候jvm会帮我们构造一个该类的实例.这么做会比较耗费资源. 如果 ...

  3. Linux_CentOS常用命令和shell命令技巧

    Linux_CentOS常用命令 关机 init 重启 init 列出当前目录的下的文件 ls //列出当前目录下的文件 ll //列出当前目录下的文件信息 等同ls -l 命令 切换目录 cd 目录 ...

  4. flutter 数据存储 SP和sqlite

    添加插件: shared_preferences: ^0.4.2 path_provider: ^1.2.0 sqflite: ^0.12.0 import 'dart:async'; import ...

  5. Android输入法遮挡了输入框,使用android:fitsSystemWindows="true"后界面顶部出现白条解决方案

    我的最外层是LinearLayout,自定义CustomLinearLayout继承LinearLayout,重写fitSystemWindows和onApplyWindowInsets两个方法: p ...

  6. jenkins结合httprunner配置实现自动化测试

    jenkins结合httprunner配置实现自动化测试 自动化测试思路: jenkins发布代码到预发布环境 –> 触发自动化测试预发布环境 à 返回测试结:如果测试通过则将代码推送到生产环境 ...

  7. window TOMCAT 端口被占用了怎么办?

    查看80端口被哪些程序占用了 netstat -ano|findstr "80" 根据pid(进程id) 查询对应的应用程序 tasklist|findstr "1828 ...

  8. python初级 1 内存和变量

    一.回顾: 1.什么是程序 一堆指令的集合 2.回想一下猜数游戏程序的特征: 1)需要输入(input) 2)会处理输入(process) 3)产生输出(output) 二.程序的一般特征:输入.处理 ...

  9. SQL Server 2014 清除用户名和密码

    网上找来找去都是SQL Server 2008版本或者以前版本的... 后来:http://stackoverflow.com/questions/349668/removing-the-rememb ...

  10. AnyCAD三维控件(转)

    在WinForm中可以方便的集成AnyCAD.Net三维控件,只需要以下几部即可完成. 一.添加DLL程序集 AnyCAD.Foundation.Net.dll AnyCAD.Presentation ...