[NEERC2007][SHOI2008]Cactus Reloaded
题目大意:
		  给你一个仙人掌,求图中相距最远的点对之间的距离。
思路:
		  Tarjan+DP。
		  我们先考虑一个树的情况。
		  设用far[u]表示点u出发到其子树中叶子节点的最大距离,若v为u的子结点,很显然far[u]=max{far[v]}+1。
		  而对于经过点u的简单路径,最长的一条肯定是max{far[v]+far[w]+2},且u≠w。
		  很显然我们只需要DFS一遍,然后随便转移即可。 
		  考虑一下仙人掌和树有什么不同。
		  很显然仙人掌就是在一棵树上加了几条边,使得图中出现了一些环,而且不会有边同时出现在两个环中。
		  我们不妨先把原图的环去掉某一个边,使得剩下的图是一棵树,很容易处理出树上的情况。
		  处理到当前环中最后一条边时,再单独对这个环进行DP。
		  考虑这个环上的每一棵外向树,设u和v是这个环上的两个结点,那么far[u]+far[v]+dis(u,v)就是一个可能的答案。
		  如何让这个答案最大化?
		  对于每一个点u,我们可以枚举每一个v来得到一个可能的答案,而要让答案尽可能大,似乎可以用单调队列来转移。
		  但唯一的问题是,现在u和v是再一个环上,他们的距离是不会单调递增的,也就是说你按顺序枚举每一个点,可能先越来越远再越来越近。
		  对于这种情况,我们把环复制一遍来转移,维护队列的时候要判断一下当前待更新的点u和用来更新的点v距离是不是超过环长的一半。
		  最后再更新一下环上高度最高的点对应的far值。
		  设环的大小为size,最高点为top,那么far[top]=max(far[top],max{far[v]+dis(top,v)})。
		  这时候要注意一下,前面DFS(Tarjan)里面,far的转移要判断一下,low[v]是不是大于等于dfn[u],如果不是,说明现在是在环上。
		  所以不能直接更新,不然后面环上DP可能会重复。
#include<queue>
#include<cstdio>
#include<cctype>
#include<vector>
inline int getint() {
register char ch;
while(!isdigit(ch=getchar()));
register int x=ch^'';
while(isdigit(ch=getchar())) x=(((x<<)+x)<<)+(ch^'');
return x;
}
const int N=;
std::vector<int> e[N];
int far[N],par[N],ans;
inline void add_edge(const int &u,const int &v) {
e[u].push_back(v);
e[v].push_back(u);
}
inline void dp(const int &top,const int &end,const int &size) {
static int cir[N*];
static std::deque<int> q;
for(register int i=size,v=end;i;i--) {
cir[size+i]=cir[i]=far[v];
v=par[v];
}
q.push_back();
for(register int i=;i<=size*;i++) {
if(i-q.front()>size/) q.pop_front();
ans=std::max(ans,cir[i]+cir[q.front()]+i-q.front());
while(!q.empty()&&cir[q.back()]-q.back()<=cir[i]-i) q.pop_back();
q.push_back(i);
}
q.clear();
for(register int i=;i<=size;i++) {
far[top]=std::max(far[top],cir[i]+std::min(i-,size-i+));
}
}
void tarjan(const int &x,const int &par) {
static int low[N],dfn[N],dep[N],cnt;
::par[x]=par;
dep[x]=dep[par]+;
low[x]=dfn[x]=++cnt;
for(unsigned i=;i<e[x].size();i++) {
const int &y=e[x][i];
if(y==par) continue;
if(!dfn[y]) {
tarjan(y,x);
low[x]=std::min(low[x],low[y]);
} else {
low[x]=std::min(low[x],dfn[y]);
}
if(dfn[x]<low[y]) {
ans=std::max(ans,far[x]+far[y]+);
far[x]=std::max(far[x],far[y]+);
}
}
for(register unsigned i=;i<e[x].size();i++) {
const int &y=e[x][i];
if(x!=::par[y]&&dfn[x]<dfn[y]) {
dp(x,y,dep[y]-dep[x]+);
}
}
}
int main() {
getint();
for(register int m=getint();m;m--) {
for(register int k=getint(),u=getint();--k;) {
const int v=getint();
add_edge(u,v);
u=v;
}
}
tarjan(,);
printf("%d\n",ans);
return ;
}
[NEERC2007][SHOI2008]Cactus Reloaded的更多相关文章
- bzoj 1023: [SHOI2008]cactus仙人掌图 tarjan缩环&&环上单调队列
		
1023: [SHOI2008]cactus仙人掌图 Time Limit: 1 Sec Memory Limit: 162 MBSubmit: 1141 Solved: 435[Submit][ ...
 - bzoj千题计划113:bzoj1023: [SHOI2008]cactus仙人掌图
		
http://www.lydsy.com/JudgeOnline/problem.php?id=1023 dp[x] 表示以x为端点的最长链 子节点与x不在同一个环上,那就是两条最长半链长度 子节点与 ...
 - bzoj 1023 [SHOI2008]cactus仙人掌图 ( poj 3567 Cactus Reloaded )——仙人掌直径模板
		
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1023 http://poj.org/problem?id=3567 因为lyd在讲课,所以有 ...
 - bzoj1023 [SHOI2008]cactus仙人掌图 & poj3567 Cactus Reloaded——求仙人掌直径
		
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1023 http://poj.org/problem?id=3567 仙人掌!直接模仿 ...
 - 1023: [SHOI2008]cactus仙人掌图 - BZOJ
		
Description如果某个无向连通图的任意一条边至多只出现在一条简单回路(simple cycle)里,我们就称这张图为仙人图(cactus).所谓简单回路就是指在图上不重复经过任何一个顶点的回路 ...
 - [SHOI2008]cactus仙人掌图
		
[题目描述] 如果某个无向连通图的任意一条边至多只出现在一条简单回路(simple cycle)里,我们就称这张图为仙人图(cactus).所谓简单回路就是指在图上不重复经过任何一个顶点的回路. 举例 ...
 - BZOJ1023:[SHOI2008]cactus仙人掌图(圆方树,DP,单调队列)
		
Description 如果某个无向连通图的任意一条边至多只出现在一条简单回路(simple cycle)里,我们就称这张图为仙人掌图(cactus). 所谓简单回路就是指在图上不重复经过任何一个顶点 ...
 - 【刷题】BZOJ 1023 [SHOI2008]cactus仙人掌图
		
Description 如果某个无向连通图的任意一条边至多只出现在一条简单回路(simple cycle)里,我们就称这张图为仙人掌图(cactus).所谓简单回路就是指在图上不重复经过任何一个顶点的 ...
 - [bzoj1023][SHOI2008]cactus 仙人掌图 (动态规划)
		
Description 如果某个无向连通图的任意一条边至多只出现在一条简单回路(simple cycle)里,我们就称这张图为仙人图(cactus).所谓简单回路就是指在图上不重复经过任何一个顶点的回 ...
 
随机推荐
- pycharm激活(JetBrains IDEA 系列产品通用xx方法(license server))
			
http://xclient.info/a/f0b9738a-36fd-8a97-a966-0d3db497092d.html .打开激活窗口 .选择 Activate new license wit ...
 - rsync安装使用详解
			
rsync是类unix系统下的数据镜像备份工具,从软件的命名上就可以看出来了——remote sync.它的特性如下: 可以镜像保存整个目录树和文件系统. 可以很容易做到保持原来文件的权限.时间.软硬 ...
 - php连接mysql报错——Fatal error: Call to undefined function mysql_connect() in
			
练习php连接mysql数据库 代码:mysql_connect("127.0.0.1:3306","root", ..... 浏览器报错:Fatal erro ...
 - 【Atcoder】ARC 080 E - Young Maids
			
[算法]数学+堆 [题意]给定n个数的排列,每次操作可以取两个数按序排在新序列的头部,求最小字典序. [题解] 转化为每次找字典序最小的两个数按序排在尾部,则p1和p2的每次选择都必须满足:p1在当前 ...
 - 【洛谷】P1648 看守 (数学)
			
题目链接 直接暴力搞\(O(n^2)\)显然是布星滴. 试想,若是一维,最远距离就是最大值减最小值. 现在推广到二维,因为有绝对值的存在,所以有四种情况 \((x1+y1) - (x2+y2), (x ...
 - 【MySQL优化】使用show status查看MySQL服务器状态信息
			
在网站开发过程中,有些时候我们需要了解MySQL的服务器状态信息,譬如当前MySQL启动后的运行时间,当前MySQL的客户端会话连接数,当前MySQL服务器执行的慢查询数,当前MySQL执行了多少SE ...
 - 自定义UINavigationController push和pop动画
			
http://segmentfault.com/q/1010000000143983 默认的UINavigationController push和pop的默认动画都是左右滑动推出,我的应用要求这种界 ...
 - 手把手教你配置苹果APNS推送服务|钿畑的博客 | 钿畑的博客
			
http://www.360doc.com/content/15/0118/17/1073512_441822850.shtml# 钿畑的文章索引 1. 什么是推送通知 2. 什么是APNS? 3. ...
 - onvif实现
			
前言 负责开发了公司的onvif,一个人从0开始写的,花了两个月 !!!下面是我的总结. onvif介绍 ONVIF[Open Network Video Interface Forum](开放型网络 ...
 - win 7 浏览器被篡改小插曲
			
今天下班回家,打开台式机发现IE,火狐都被篡改了.作为运维都会有点强迫症.这是个桌面系统,实在是没兴趣捣鼓.但是还是没办法,经常要用.等我下次有空了,直接换linux好了. 于是开始排查问题吧: 1. ...