学习了一下圆方树。

圆方树是一种可以处理仙人掌的数据结构,具体见这里:http://immortalco.blog.uoj.ac/blog/1955

简单来讲它是这么做的:用tarjan找环,然后对每个环建立一个新点,然后将环上的边删去,并环上的每个点都连到新点上。这样我们就可以把一个环缩成一个菊花图,重复这么做,一棵仙人掌就变成一棵树啦!这棵树就叫做圆方树,其中原点叫圆点,新点叫方点。

圆方树和原仙人掌很相似,而且它又是一棵树,于是我们就可以在上面dp啦!不过要注意的是对于方点的处理,不能直接更新,要作一个单调队列,因为环上有两种走法么。

期间调了很久,而且我还不会生成数据,orz cbh。

tarjan求DCC都不会写了,我好弱啊~

#include <bits/stdc++.h>
#define N 110000
using namespace std;
int n, m, nn;
vector <int> V[N], W[N];
int low[N], dfn[N], dfsnum;
int f[N];
int ans;
int shed[N], top;
deque <int> Q;
void tarjan(int t, int f)
{
dfn[t] = low[t] = ++ dfsnum;
shed[++ top] = t;
for (int p = ; p < V[t].size(); ++ p)
if (V[t][p] != f) if (!dfn[V[t][p]])
{
tarjan(V[t][p], t);
low[t] = min(low[t], low[V[t][p]]);
if (low[V[t][p]] >= dfn[t])
{
if (shed[top] != V[t][p])
{
nn ++;
int cur;
do
{
//W[shed[top]].push_back(nn);
W[nn].push_back(cur = shed[top]);
shed[top --] = ;
}
while (cur != V[t][p]);
W[t].push_back(nn);
//W[nn].push_back(t);
}
else
{
W[t].push_back(shed[top]);
//W[shed[top]].push_back(t);
shed[top --] = ;
}
}
}
else low[t] = min(low[t], dfn[V[t][p]]);
}
void dfs(int t)
{
f[t] = ;
int s = W[t].size();
if (t > n)
{
for (int p = ; p < s; ++ p)
dfs(W[t][p]);
for (int q = ; q < (s + ) / ; ++ q)
{
while (!Q.empty() && f[W[t][Q.back()]] + Q.back() < f[W[t][q]] + q) Q.pop_back();
Q.push_back(q);
}
for (int p = , q = (s + ) / ; p < s; ++ p, q = (q == s? : q + ) )
{
if (q != s)
{
while (!Q.empty() && f[W[t][Q.back()]] + (Q.back() < p? Q.back() + s + - p: Q.back() - p) < f[W[t][q]] + (q < p? q + s + - p: q - p)) Q.pop_back();
Q.push_back(q);
}
if (Q.front() == p) Q.pop_front();
if (!Q.empty()) ans = max(ans, f[W[t][p]] + f[W[t][Q.front()]] + (Q.front() > p? Q.front() - p: Q.front() + s + - p));
}
for (int p = ; p < (s + ) / ; ++ p) f[t] = max(f[t], f[W[t][p]] + p);
for (int p = (s + ) / ; p < s; ++ p) f[t] = max(f[t], f[W[t][p]] + s - - p);
Q.clear();
}
else
{
int mx1 = , mx2 = ;
for (int p = ; p < s; ++ p)
{
dfs(W[t][p]);
if (f[W[t][p]] + > mx1) mx2 = mx1, mx1 = f[W[t][p]] + ;
else if (f[W[t][p]] + > mx2) mx2 = f[W[t][p]] + ;
}
ans = max(ans, mx1 + mx2);
f[t] = mx1;
}
}
int main()
{
scanf("%d%d", &n, &m);
for (int i = ; i <= m; ++ i)
{
int k, a;
scanf("%d%d", &k, &a);
for (int i = ; i <= k; ++ i)
{
int b;
scanf("%d", &b);
V[a].push_back(b);
V[b].push_back(a);
a = b;
}
}
nn = n;
tarjan(, );
//puts("haha");
dfs();
printf("%d\n", ans);
}

bzoj1023: [SHOI2008]cactus仙人掌图的更多相关文章

  1. bzoj千题计划113:bzoj1023: [SHOI2008]cactus仙人掌图

    http://www.lydsy.com/JudgeOnline/problem.php?id=1023 dp[x] 表示以x为端点的最长链 子节点与x不在同一个环上,那就是两条最长半链长度 子节点与 ...

  2. BZOJ1023:[SHOI2008]cactus仙人掌图(圆方树,DP,单调队列)

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

  3. BZOJ1023: [SHOI2008]cactus仙人掌图(仙人掌dp)

    Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 3467  Solved: 1438[Submit][Status][Discuss] Descripti ...

  4. BZOJ1023[SHOI2008]cactus仙人掌图 【仙人掌DP】

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

  5. BZOJ1023: [SHOI2008]cactus仙人掌图(仙人掌)

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

  6. [bzoj1023][SHOI2008]cactus 仙人掌图 (动态规划)

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

  7. 2018.10.29 bzoj1023: [SHOI2008]cactus仙人掌图(仙人掌+单调队列优化dp)

    传送门 求仙人掌的直径. 感觉不是很难. 分点在环上面和不在环上分类讨论. 不在环上直接树形dpdpdp. 然后如果在环上讨论一波. 首先对环的祖先有贡献的只有环上dfsdfsdfs序最小的点. 对答 ...

  8. bzoj1023 [SHOI2008]cactus仙人掌图 & poj3567 Cactus Reloaded——求仙人掌直径

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1023    http://poj.org/problem?id=3567 仙人掌!直接模仿 ...

  9. bzoj千题计划224:bzoj1023: [SHOI2008]cactus仙人掌图

    又写了一遍,发出来做个记录 #include<cstdio> #include<algorithm> #include<iostream> using namesp ...

随机推荐

  1. 面试题:return和finally执行

    创建一个包含return和finally的方法:(如下所示) public class Demo { public int get() { int x=1; try { x++; return x; ...

  2. AC日记——中位数 洛谷 P1168

    题目描述 给出一个长度为N的非负整数序列A[i],对于所有1 ≤ k ≤ (N + 1) / 2,输出A[1], A[2], …, A[2k - 1]的中位数.[color=red]即[/color] ...

  3. centos 6x系统下源码安装mysql操作记录

    在运维工作中经常部署各种运维环境,涉及mysql数据库的安装也是时常需要的.mysql数据库安装可以选择yum在线安装,但是这种安装的mysql一般是系统自带的,版本方面可能跟需求不太匹配.可以通过源 ...

  4. Recover data from reference electrode via EEGLab 用EEGLab恢复参考电极数据

    The data of scanning reference electrode will not show initially. Here is a summary of recovering it ...

  5. 使用对话框 —— Dialog

      对话框就是一般的弹出窗口,主要用来提示用户,和用户交互.   创建Activity对话框 使用Activity模拟对话框.这个比较简单,主要是使用Activity自带的Dialog主题.   创建 ...

  6. php基础知识整理

    记录一些php容易忽略的基础知识点 include和require的区别 require和include都表示引入指定文件,主要区别有几点 1.加载失败处理方式不同  include在引入不存文件时产 ...

  7. Vue.js之v-if

    ---恢复内容开始--- 首先,在跟着api做的时候,要把v-if包裹在Vue对应的el里面,重要的事情说三遍. html: <div id="vif"> <di ...

  8. Unable to load R3 module D:\Program Files\Oracle\VirtualBox/VBoxDD.DLL (VBoxDD): GetLastError=1790 (VERR_UNRESOLVED_ERROR).

    Unable to load R3 module D:\Program Files\Oracle\VirtualBox/VBoxDD.DLL (VBoxDD): GetLastError=1790 ( ...

  9. gulp删除文件和文件夹

    使用模块:del,npm install --save-dev gulp del var gulp = require('gulp'); var del = require('del'); gulp. ...

  10. hadoop在网页客户端的maven配置

    hadoop网页客户端maven配置,只能在tomcat7上运行,tomcat6和tomcat8运行会出错,我用的是tomcat-7.0.67 完整的pom.xml内容为: <!-- 这个配置只 ...