Portal -->bzoj2500

Description

​   给你一棵树,每条边有边权,有两个给给的人第\(i\)天会从编号为\(i\)的点出发走这个点的树上最长距离,现在要你求一个最长的\(len\)满足\(dis_{st},dis_{st+1}...dis_{st+len-1}\)满足其中的最大值最小值之差不大于\(m\),\(dis_{i}\)表示第\(i\)天走的距离,\(st\)不一定为\(1\)

Solution

​   很好这题。。我一开始看错题了以为是一个弱智题(看成相邻两个差\(<=m\)了。。)

​   然后我没有看空间十分开心写了一个预处理rmq+双指针乱搞的玩意==

​​   后来终于走回正道写了单调队列。。没救了真的

​​   感觉自己对单调队列的运用还是不够熟练,所以还是搬上来加深一下印象好了

​ ​  

​   这题首先求每个点为起点的树上最长距离。。经典树形dp维护子树内最大值最小值可以\(O(n)\)求出存在\(dis\)数组里面

​​   然后我们考虑用这样的方式求一个最长符合条件区间:我们考虑将\(n\)个数依次加进当前的区间中,记现在加到第\(i\)个数,然后在\(i-1\)这个位置结尾的最长符合条件区间的左端点为\(st\),那么将\(dis[i]\)加进来的话,如果此时区间内最大值最小值符合条件,那么更新答案,否则我们需要调整区间的左端点,将其移到一个最靠左的满足原来的\(min\)或者\(max\)不在新区间内的位置(也就是\(min\)和\(max\)中最靠前的那个的位置\(+1\)),这样我们就可以得到以每个\(i\)为右端点的最长区间,\(ans\)必定为其中的最大值

​   那么我们只要开两个双端队列维护当前区间内的最大值和最小值就好了,维护最大值的队列保持单调递减,维护最小值的队列保持单调递增,每次需要调整区间的时候只要找队头中较靠左的位置\(+1\)并将\(st\)调到这个位置即可

​  

​​   代码大概长这个样子

  1. #include<iostream>
  2. #include<cstdio>
  3. #include<cstring>
  4. #include<algorithm>
  5. #include<cmath>
  6. #define ll long long
  7. using namespace std;
  8. const int N=1000010,TOP=20;
  9. struct xxx{
  10. int y,nxt;
  11. ll dis;
  12. }a[N];
  13. struct Data{/*{{{*/
  14. int which;
  15. ll mx,smx;
  16. Data(){}
  17. Data(int _which,ll _mx,ll _smx){which=_which; mx=_mx; smx=_smx;}
  18. void update(int u,ll d){
  19. if (mx<d)
  20. smx=mx,which=u,mx=d;
  21. else
  22. smx=max(smx,d);
  23. }
  24. }info[N];/*}}}*/
  25. int mnq[N],mxq[N];
  26. ll dis[N];
  27. int h[N];
  28. int l1,r1,l2,r2;
  29. int n,m,tot,ans;
  30. void add(int x,int y,ll d){a[++tot].y=y; a[tot].nxt=h[x]; h[x]=tot; a[tot].dis=d;}
  31. void dfs(int x,ll d){
  32. int u;
  33. info[x]=Data(x,0,0); dis[x]=0;
  34. for (int i=h[x];i!=-1;i=a[i].nxt){
  35. u=a[i].y;
  36. dfs(u,d+a[i].dis);
  37. info[x].update(u,info[u].mx+a[i].dis);
  38. dis[x]=max(dis[u]+a[i].dis,dis[x]);
  39. }
  40. }
  41. void dfs1(int fa,int x,ll predis){
  42. int u;
  43. if (fa){
  44. if (x==info[fa].which){
  45. dis[x]=max(dis[x],info[fa].smx+predis);
  46. info[x].update(fa,info[fa].smx+predis);
  47. }
  48. else{
  49. dis[x]=max(dis[x],info[fa].mx+predis);
  50. info[x].update(fa,info[fa].mx+predis);
  51. }
  52. }
  53. for (int i=h[x];i!=-1;i=a[i].nxt){
  54. u=a[i].y;
  55. dfs1(x,u,a[i].dis);
  56. }
  57. }
  58. void solve(){
  59. int st=1;
  60. l1=l2=1; r1=r2=0;
  61. for (int i=1;i<=n;++i){
  62. while (r1>=l1&&dis[mnq[r1]]>=dis[i]) --r1;
  63. mnq[++r1]=i;
  64. while (r2>=l2&&dis[mxq[r2]]<=dis[i]) --r2;
  65. mxq[++r2]=i;
  66. while (dis[mxq[l2]]-dis[mnq[l1]]>m){
  67. if (mnq[l1]<mxq[l2])
  68. st=mnq[l1]+1,++l1;
  69. else
  70. st=mxq[l2]+1,++l2;
  71. }
  72. ans=max(ans,i-st+1);
  73. }
  74. }
  75. int main(){
  76. #ifndef ONLINE_JUDGE
  77. freopen("a.in","r",stdin);
  78. #endif
  79. ll d;
  80. int fa;
  81. scanf("%d%d",&n,&m);
  82. memset(h,-1,sizeof(h));
  83. tot=0;
  84. for (int i=2;i<=n;++i){
  85. scanf("%d%lld",&fa,&d);
  86. add(fa,i,d);
  87. }
  88. dfs(1,0);
  89. dfs1(0,1,0);
  90. solve();
  91. printf("%d\n",ans);
  92. }

【bzoj2500】幸福的道路的更多相关文章

  1. BZOJ2500: 幸福的道路

    题解: 一道不错的题目. 树DP可以求出从每个点出发的最长链,复杂度O(n) 然后就变成找一个数列里最长的连续区间使得最大值-最小值<=m了. 成了这题:http://www.cnblogs.c ...

  2. bzoj2500幸福的道路 树形dp+单调队列

    2500: 幸福的道路 Time Limit: 20 Sec  Memory Limit: 256 MBSubmit: 434  Solved: 170[Submit][Status][Discuss ...

  3. [Bzoj2500]幸福的道路(树上最远点)

    2500: 幸福的道路 Time Limit: 20 Sec  Memory Limit: 256 MBSubmit: 474  Solved: 194[Submit][Status][Discuss ...

  4. bzoj2500: 幸福的道路(树形dp+单调队列)

    好题.. 先找出每个节点的树上最长路 由树形DP完成 节点x,设其最长路的子节点为y 对于y的最长路,有向上和向下两种情况: down:y向子节点的最长路g[y][0] up:x的次长路的g[x][1 ...

  5. 【BZOJ2500】幸福的道路 树形DP+RMQ+双指针法

    [BZOJ2500]幸福的道路 Description 小T与小L终于决定走在一起,他们不想浪费在一起的每一分每一秒,所以他们决定每天早上一同晨练来享受在一起的时光. 他们画出了晨练路线的草图,眼尖的 ...

  6. 【BZOJ】【2500】幸福的道路

    树形DP+单调队列优化DP 好题(也是神题……玛雅我实在是太弱了TAT,真是一个250) 完全是抄的zyf的……orz我还是退OI保平安吧 第一步对于每一天求出一个从第 i 个点出发走出去的最长链的长 ...

  7. [BZOJ 2500] 幸福的道路

    照例先贴题面(汪汪汪) 2500: 幸福的道路 Time Limit: 20 Sec  Memory Limit: 256 MBSubmit: 368  Solved: 145[Submit][Sta ...

  8. 【bzoj2500】幸福的道路 树形dp+单调队列

    Description 小T与小L终于决定走在一起,他们不想浪费在一起的每一分每一秒,所以他们决定每天早上一同晨练来享受在一起的时光. 他们画出了晨练路线的草图,眼尖的小T发现可以用树来描绘这个草图. ...

  9. 【bzoj2500】幸福的道路 树形dp+倍增RMQ+二分

    原文地址:http://www.cnblogs.com/GXZlegend/p/6825389.html 题目描述 小T与小L终于决定走在一起,他们不想浪费在一起的每一分每一秒,所以他们决定每天早上一 ...

  10. (noip模拟二十一)【BZOJ2500】幸福的道路-树形DP+单调队列

    Description 小T与小L终于决定走在一起,他们不想浪费在一起的每一分每一秒,所以他们决定每天早上一同晨练来享受在一起的时光. 他们画出了晨练路线的草图,眼尖的小T发现可以用树来描绘这个草图. ...

随机推荐

  1. VS2013只显示会附加到进程,无法启动调试

    今天在使用VS2013的时候,打开突然发现,只显示附加到进程,无法进行调试,调试位置显示灰色,到网上各处寻求答案,本以为是个大问题,没想到只是个小问题.主要原因只是后台开太多东西了,导致VS2013运 ...

  2. datax 执行流程分析

    https://www.jianshu.com/nb/29319571 https://www.jianshu.com/p/b10fbdee7e56

  3. Nginx内容缓存

    本节介绍如何启用和配置从代理服务器接收的响应的缓存.主要涉及以下内容 - 缓存介绍 启用响应缓存 涉及缓存的NGINX进程 指定要缓存的请求 限制或绕过缓存 从缓存中清除内容 配置缓存清除 发送清除命 ...

  4. website for personal research

    YOLO https://pjreddie.com/darknet/yolo/ Low Rank Matrix Recovery and Completion via Convex Optimizat ...

  5. KETTLE设置变量

    一.kettle变量类型 kettle变量分为: 1.环境变量 通过 set variables组件设置变量,范围可以是:JVM变量.作业变量.父作业变量.根作业变量.使用时通过${var}或 %%v ...

  6. c++ Dynamic Memory (part 2)

    Don't use get to initialize or assign another smart pointer. The code that use the return from get c ...

  7. [C++] Solve "Cannot run program "gdb": Unknown reason" error

    In Mac OSX, The Issue Image: 1. Build the project on Eclipse successfully. 2. Run gdb on command lin ...

  8. php新手需要注意的高效率编程

    1.尽量静态化: 如果一个方法能被静态,那就声明它为静态的,速度可提高1/4,甚至我测试的时候,这个提高了近三倍.   当然了,这个测试方法需要在十万级以上次执行,效果才明显.   其实静态方法和非静 ...

  9. HDU 5207 Greatest Greatest Common Divisor

    题目链接: hdu:http://acm.hust.edu.cn/vjudge/problem/visitOriginUrl.action?id=153598 bc(中文):http://bestco ...

  10. C#高级编程 (第六版) 学习 第六章:运算符和类型强制转换

    第六章 运算符和类型强制转换 1,运算符 类别 运算符 算术运算符 + - * / % 逻辑运算符 & | ^ ~ && || ! 字符串连接运算符 + 增量和减量运算符 ++ ...