本来想先求出点双再一个一个处理结果写了很长发现太麻烦

设f[u]为u点向下的最长链

就是再tarjan的过程中,先照常处理,用最长儿子链和次长儿子链更新按ans,然后处理以这个点为根的环,也就是这个点是dfs第一次到这个环访问到的点

环用来更新ans的是儿子链+到根的一段,这个直接for一边就行,还有就是一个儿子链+环上的一段+另一个儿子链,这个把环复制一遍然后单调队列扫即可

注意距离的定义是最短距离!

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

bzoj 1023: [SHOI2008]cactus仙人掌图【tarjan+dp+单调队列】的更多相关文章

  1. bzoj 1023: [SHOI2008]cactus仙人掌图 tarjan缩环&&环上单调队列

    1023: [SHOI2008]cactus仙人掌图 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 1141  Solved: 435[Submit][ ...

  2. BZOJ.1023.[SHOI2008]cactus仙人掌图(DP)

    题目链接 类似求树的直径,可以用(类似)树形DP求每个点其子树(在仙人掌上就是诱导子图)最长链.次长链,用每个点子节点不同子树的 max{最长链}+max{次长链} 更新答案.(不需要存次长链,求解过 ...

  3. 【刷题】BZOJ 1023 [SHOI2008]cactus仙人掌图

    Description 如果某个无向连通图的任意一条边至多只出现在一条简单回路(simple cycle)里,我们就称这张图为仙人掌图(cactus).所谓简单回路就是指在图上不重复经过任何一个顶点的 ...

  4. bzoj 1023: [SHOI2008]cactus仙人掌图 2125: 最短路 4728: 挪威的森林 静态仙人掌上路径长度的维护系列

    %%% http://immortalco.blog.uoj.ac/blog/1955 一个通用的写法是建树,对每个环建一个新点,去掉环上的边,原先环上每个点到新点连边,边权为点到环根的最短/长路长度 ...

  5. bzoj 1023: [SHOI2008]cactus仙人掌图

    这道题是我做的第一道仙人掌DP,小小纪念一下…… 仙人掌DP就是环上的点环状DP,树上的点树上DP.就是说,做一遍DFS,DFS的过程中处理出环,环上的点先不DP,先把这些换上的点的后继点都处理出来, ...

  6. BZOJ 1023: [SHOI2008]cactus仙人掌图 | 在仙人掌上跑DP

    题目: 求仙人掌直径 http://www.lydsy.com/JudgeOnline/problem.php?id=1023 题解: 首先给出仙人掌的定义:满足所有的边至多在一个环上的无向联通图 我 ...

  7. bzoj 1023 [SHOI2008]cactus仙人掌图 ( poj 3567 Cactus Reloaded )——仙人掌直径模板

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1023 http://poj.org/problem?id=3567 因为lyd在讲课,所以有 ...

  8. 1023: [SHOI2008]cactus仙人掌图 - BZOJ

    Description如果某个无向连通图的任意一条边至多只出现在一条简单回路(simple cycle)里,我们就称这张图为仙人图(cactus).所谓简单回路就是指在图上不重复经过任何一个顶点的回路 ...

  9. 1023: [SHOI2008]cactus仙人掌图(DP+单调队列优化)

    这道题吗= =首先解决了我多年以来对仙人掌图的疑问,原来这种高大上的东西原来是这个啊= = 然后,看到这种题,首先必须的就是缩点= = 缩点完之后呢,变成在树上找最长路了= =直接树形dp了 那么那些 ...

随机推荐

  1. LeetCode ||& Word Break && Word Break II(转)——动态规划

    一. Given a string s and a dictionary of words dict, determine if s can be segmented into a space-sep ...

  2. Android兼容性测试CTS

     一.简介 为了确保Android应用能够在所有兼容Android的设备上正确运行,并且保持相似的用户体验,在每个版本发布之时,Android提供了一套兼容性测试用例集合(Compatibility ...

  3. oracle 11G direct path read 非常美也非常伤人

    direct path read 在11g中,全表扫描可能使用direct path read方式,绕过buffer cache,这种全表扫描就是物理读了. 在10g中,都是通过gc buffer来读 ...

  4. (转载)js(jquery)的on绑定点击事件执行两次的解决办法

    js(jquery)的on绑定点击事件执行两次的解决办法—不是事件绑定而是事件冒泡 遇到的问题:jquery中用.on()给页面中新加的元素添加点击事件时,点击事件源,绑定的事件执行两次,这里的ale ...

  5. Ajax的简单实现(Json)

    之前写的是一般的Ajax if (request.status === 200) { document.getElementById("createResult").innerHT ...

  6. jquery一个比较好的轮播图jQuery.kinMaxShow介绍

    kinMaxShow API 可选参数以及详解 kinMaxShow 主参数详解 参数名称 默认值 简单释义 height 500 [整型 (单位:像素)]焦点图高度,必须设置 缺省则启用默认高度 5 ...

  7. querying rpm database

    Call dbMatch on a transaction set to create a match iterator. As with the C API, a match iterator al ...

  8. a column-oriented DBMS

    https://clickhouse.yandex/docs/en/introduction/what_is_clickhouse.html

  9. 集群环境搭建-SSH免密码登陆(二)

    1.打开sshd配置 命令: vi /etc/ssh/sshd_config 找到以下内容,并去掉注释符”#“ RSAAuthentication yes PubkeyAuthentication y ...

  10. (linux)SD卡初始化-mmc_sd_init_card函数

      为了学习SD/SDIO协议,看了一下linux中初始化SD卡的流程,结合代码更容易SD初始化是怎么做的. 下面图截自:"SD Specifications Part 1 Physical ...