我太菜了居然调了一上午……

这个题就是要求基环树森林的基环树直径和

大概步骤就是找环—>dp找每个环点最远能到达距离作为点权—>复制一倍环,单调队列dp

找环是可以拓扑的,但是利用性质有更快好写的做法,就是像朱刘算法找环那样,按照输入的方向(i—>to_i)打一圈标记,如果碰到同样标记就说明有环,这里注意我一开始没注意到的,从i点进入找到环不代表i点在环上,因为可能是6字形的,所以一定是环点的是找到的有同样标记的那个点,然后顺着这个点把环点都放进一个栈(其实不用,但是这样好写一些),顺着每个点向非环点dfs找到最远点,这里注意!非常坑的是一棵基环树的直径不一定经过环,所以在dfs过程中,把每个点向下的最长和次长路径的和都和一个全局变量mx取max。把每个点向下最长当做点权va[i]

这样就只剩一个环长为len的环了,把这个环复制一遍,sum表示长度前缀和,答案就是max(va[i]+va[j]+sum[j]-sum[i])(j-i<len),这个可以用单调队列实现,能得到答案ans

这样,max(ans,mx)就是当前这棵基环树的直径了

把每次找到的环的答案加起来即可

  1. #include<iostream>
  2. #include<cstdio>
  3. using namespace std;
  4. const int N=1000005;
  5. int n,ne[N],le[N],h[N],cnt,v[N],s[N],top,q[N];
  6. long long l[N],va[N<<1],ans,sm[N<<1],mxx;
  7. struct qwe
  8. {
  9. int ne,to,va;
  10. }e[N<<1];
  11. int read()
  12. {
  13. int r=0,f=1;
  14. char p=getchar();
  15. while(p>'9'||p<'0')
  16. {
  17. if(p=='-')
  18. f=-1;
  19. p=getchar();
  20. }
  21. while(p>='0'&&p<='9')
  22. {
  23. r=r*10+p-48;
  24. p=getchar();
  25. }
  26. return r*f;
  27. }
  28. void add(int u,int v,int w)
  29. {
  30. cnt++;
  31. e[cnt].ne=h[u];
  32. e[cnt].to=v;
  33. e[cnt].va=w;
  34. h[u]=cnt;
  35. }
  36. long long dfs(int u,int fa,int f)
  37. {
  38. long long mx1=0,mx2=0;
  39. for(int i=h[u];i;i=e[i].ne)
  40. if(v[e[i].to]!=-1&&e[i].to!=fa)
  41. {
  42. long long w=dfs(e[i].to,u,f)+e[i].va;
  43. if(w>mx1)
  44. mx2=mx1,mx1=w;
  45. else if(w>mx2)
  46. mx2=w;
  47. }
  48. mxx=max(mxx,mx1+mx2);
  49. return mx1;
  50. }
  51. long long dp()
  52. {
  53. for(int i=1;i<=top;i++)
  54. sm[i]=sm[i-1]+l[i];
  55. for(int i=top+1;i<=top*2;i++)
  56. sm[i]=sm[i-1]+l[i-top],va[i]=va[i-top];
  57. int l=1,r=1;q[1]=1;
  58. long long ans=0;
  59. for(int i=2;i<=top*2;i++)
  60. {
  61. while(l<r&&i-q[l]>=top)
  62. l++;
  63. ans=max(ans,sm[i]+va[i]-(sm[q[l]]-va[q[l]]));
  64. while(l<r&&sm[q[r]]-va[q[r]]>sm[i]-va[i])
  65. r--;
  66. q[++r]=i;
  67. }
  68. return ans;
  69. }
  70. int main()
  71. {
  72. n=read();
  73. for(int i=1;i<=n;i++)
  74. {
  75. ne[i]=read(),le[i]=read();
  76. add(i,ne[i],le[i]),add(ne[i],i,le[i]);
  77. }
  78. for(int i=1,j;i<=n;i++)
  79. if(!v[i])
  80. {
  81. top=0;
  82. for(j=i;!v[j];j=ne[j])
  83. v[j]=i;
  84. if(v[j]!=i)
  85. continue;
  86. s[++top]=j;
  87. for(int u=ne[j];u!=j;u=ne[u])
  88. s[++top]=u;
  89. for(j=1;j<=top;j++)
  90. v[s[j]]=-1;
  91. for(j=1;j<=top;j++)
  92. l[j+1]=le[s[j]];
  93. l[1]=l[top+1];
  94. long long mx=0;
  95. for(j=1;j<=top;j++)
  96. {
  97. mxx=0;
  98. va[j]=dfs(s[j],0,i);
  99. mx=max(mx,mxx);
  100. }
  101. ans+=max(dp(),mx);
  102. for(j=1;j<=top;j++)
  103. v[s[j]]=i;
  104. }
  105. printf("%lld\n",ans);
  106. return 0;
  107. }

bzoj 1791: [Ioi2008]Island 岛屿【基环树+单调队列优化dp】的更多相关文章

  1. BZOJ1791 [Ioi2008]Island 岛屿[基环树+单调队列优化DP]

    基环树直径裸题. 首先基环树直径只可能有两种形式:每棵基环树中的环上挂着的树的直径,或者是挂在环上的两个树的最大深度根之间的距离之和. 所以,先对每个连通块跑一遍,把环上的点找出来,然后对环上每个点跑 ...

  2. P4381 [IOI2008]Island(基环树+单调队列优化dp)

    P4381 [IOI2008]Island 题意:求图中所有基环树的直径和 我们对每棵基环树分别计算答案. 首先我们先bfs找环(dfs易爆栈) 蓝后我们处理直径 直径不在环上,就在环上某点的子树上 ...

  3. BZOJ 1791: [IOI2008]Island 岛屿 - 基环树

    传送门 题解 题意 = 找出无向基环树森林的每颗基环树的直径. 我们首先需要找到每颗基环树的环, 但是因为是无向图,用tarjan找环, 加个手工栈, 我也是看了dalao的博客才知道tarjan找无 ...

  4. [bzoj1791][ioi2008]Island 岛屿(基环树、树的直径)

    [bzoj1791][ioi2008]Island 岛屿(基环树.树的直径) bzoj luogu 题意可能会很绕 一句话:基环树的直径. 求直径: 对于环上每一个点记录其向它的子树最长路径为$dp_ ...

  5. BZOJ 2806: [Ctsc2012]Cheat [广义后缀自动机 单调队列优化DP 二分]

    2806: [Ctsc2012]Cheat 题意: 多个主串和多个询问串,每次询问将询问串分成多个连续子串,如果一个子串长度>=L且在主串中出现过就是熟悉的 如果熟悉的字符串长度>=询问串 ...

  6. bzoj 2806 [Ctsc2012]Cheat——广义后缀自动机+单调队列优化DP

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2806 只想着怎么用后缀数据结构做,其实应该考虑结合其他算法. 可以二分那个长度 L .设当前 ...

  7. BZOJ 2442 [Usaco2011 Open]修剪草坪:单调队列优化dp

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=2442 题意: 有n个数a[i]从左到右排成一排. 你可以任意选数,但是连续的数不能超过k个 ...

  8. bzoj 1791: [Ioi2008]Island 岛屿

    #include<iostream> #include<cstdio> #define M 1000009 using namespace std; *M],cnt,n,hea ...

  9. BZOJ1791[Ioi2008]Island 岛屿 ——基环森林直径和+单调队列优化DP+树形DP

    题目描述 你将要游览一个有N个岛屿的公园.从每一个岛i出发,只建造一座桥.桥的长度以Li表示.公园内总共有N座桥.尽管每座桥由一个岛连到另一个岛,但每座桥均可以双向行走.同时,每一对这样的岛屿,都有一 ...

随机推荐

  1. 笔记——python风格规范

    分号 不要在行尾加分号, 也不要用分号将两条命令放在同一行. 行长度 每行不超过80个字符 例外: 长的导入模块语句 注释里的URL 不要使用反斜杠连接行. Python会将 圆括号, 中括号和花括号 ...

  2. JAVA IO中read()方法的返回值

    read()方法的作用是从输入流读取数据的下一个字节,返回的字节的值是一个0~255之间的整数.到达流的末尾返回-1. 刚开始我以为这个返回值表示的是所读取的数据的长度,可是之后在一个示例程序中发现这 ...

  3. HDU1166 线段树裸题 区间求和

    敌兵布阵 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submis ...

  4. hdu - 2822 Dogs (优先队列+bfs)

    http://acm.hdu.edu.cn/showproblem.php?pid=2822 给定起点和终点,问从起点到终点需要挖几次只有从# 到 .或者从. 到  . 才需要挖一次. #includ ...

  5. mysql-performance-schema

    http://www.fromdual.com/mysql-performance-schema-hints http://www.cnblogs.com/cchust/

  6. 纤程(FIBER)

    Indy 10 还包含对纤程的支持.纤程是什么?简单来说,它也是 一个“线程”,但是它是由代码控制的,而不是由操作系统控制的.实际上,可以认为线程 是一个高级纤程.纤程和 Unix 用户线程(Unix ...

  7. android账号与同步之账号管理

    在android提供的sdk中,samples文件夹下有一个叫SampleSyncAdapter的演示样例,它是一个账号与同步的实例,比方Google原始的android手机能够使用Google账号进 ...

  8. script标签async和defer的区别及作用

    作用: 1.没有 defer 或 async,浏览器会立即加载并执行指定的脚本,也就是说不等待后续载入的文档元素,读到就加载并执行. 2.async 属性表示异步执行引入的 JavaScript,与 ...

  9. 关于MacBook怎么更新Android SDK

    昨天公司的人给了我一个VPN,可是还是无法更新SDK,后来发现将下图: 通过VPN发送全部流量勾选以后就能够连接更新了,哎.处处皆学问,特此分享一下此经验. 喜欢的朋友关注我哦! 多谢支持

  10. Cocos2d-x3.1下 Android,APK自己主动升级

    项目要做Android的自己主动升级,对于我们之前做iOS的转Cocos开发做Android方面的功能..... 不正确说了.这里记录下我的实现过程. 原文地址:http://blog.csdn.ne ...