bzoj 1791: [Ioi2008]Island 岛屿【基环树+单调队列优化dp】
我太菜了居然调了一上午……
这个题就是要求基环树森林的基环树直径和
大概步骤就是找环—>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)就是当前这棵基环树的直径了
把每次找到的环的答案加起来即可
#include<iostream>
#include<cstdio>
using namespace std;
const int N=1000005;
int n,ne[N],le[N],h[N],cnt,v[N],s[N],top,q[N];
long long l[N],va[N<<1],ans,sm[N<<1],mxx;
struct qwe
{
int ne,to,va;
}e[N<<1];
int read()
{
int r=0,f=1;
char p=getchar();
while(p>'9'||p<'0')
{
if(p=='-')
f=-1;
p=getchar();
}
while(p>='0'&&p<='9')
{
r=r*10+p-48;
p=getchar();
}
return r*f;
}
void add(int u,int v,int w)
{
cnt++;
e[cnt].ne=h[u];
e[cnt].to=v;
e[cnt].va=w;
h[u]=cnt;
}
long long dfs(int u,int fa,int f)
{
long long mx1=0,mx2=0;
for(int i=h[u];i;i=e[i].ne)
if(v[e[i].to]!=-1&&e[i].to!=fa)
{
long long w=dfs(e[i].to,u,f)+e[i].va;
if(w>mx1)
mx2=mx1,mx1=w;
else if(w>mx2)
mx2=w;
}
mxx=max(mxx,mx1+mx2);
return mx1;
}
long long dp()
{
for(int i=1;i<=top;i++)
sm[i]=sm[i-1]+l[i];
for(int i=top+1;i<=top*2;i++)
sm[i]=sm[i-1]+l[i-top],va[i]=va[i-top];
int l=1,r=1;q[1]=1;
long long ans=0;
for(int i=2;i<=top*2;i++)
{
while(l<r&&i-q[l]>=top)
l++;
ans=max(ans,sm[i]+va[i]-(sm[q[l]]-va[q[l]]));
while(l<r&&sm[q[r]]-va[q[r]]>sm[i]-va[i])
r--;
q[++r]=i;
}
return ans;
}
int main()
{
n=read();
for(int i=1;i<=n;i++)
{
ne[i]=read(),le[i]=read();
add(i,ne[i],le[i]),add(ne[i],i,le[i]);
}
for(int i=1,j;i<=n;i++)
if(!v[i])
{
top=0;
for(j=i;!v[j];j=ne[j])
v[j]=i;
if(v[j]!=i)
continue;
s[++top]=j;
for(int u=ne[j];u!=j;u=ne[u])
s[++top]=u;
for(j=1;j<=top;j++)
v[s[j]]=-1;
for(j=1;j<=top;j++)
l[j+1]=le[s[j]];
l[1]=l[top+1];
long long mx=0;
for(j=1;j<=top;j++)
{
mxx=0;
va[j]=dfs(s[j],0,i);
mx=max(mx,mxx);
}
ans+=max(dp(),mx);
for(j=1;j<=top;j++)
v[s[j]]=i;
}
printf("%lld\n",ans);
return 0;
}
bzoj 1791: [Ioi2008]Island 岛屿【基环树+单调队列优化dp】的更多相关文章
- BZOJ1791 [Ioi2008]Island 岛屿[基环树+单调队列优化DP]
基环树直径裸题. 首先基环树直径只可能有两种形式:每棵基环树中的环上挂着的树的直径,或者是挂在环上的两个树的最大深度根之间的距离之和. 所以,先对每个连通块跑一遍,把环上的点找出来,然后对环上每个点跑 ...
- P4381 [IOI2008]Island(基环树+单调队列优化dp)
P4381 [IOI2008]Island 题意:求图中所有基环树的直径和 我们对每棵基环树分别计算答案. 首先我们先bfs找环(dfs易爆栈) 蓝后我们处理直径 直径不在环上,就在环上某点的子树上 ...
- BZOJ 1791: [IOI2008]Island 岛屿 - 基环树
传送门 题解 题意 = 找出无向基环树森林的每颗基环树的直径. 我们首先需要找到每颗基环树的环, 但是因为是无向图,用tarjan找环, 加个手工栈, 我也是看了dalao的博客才知道tarjan找无 ...
- [bzoj1791][ioi2008]Island 岛屿(基环树、树的直径)
[bzoj1791][ioi2008]Island 岛屿(基环树.树的直径) bzoj luogu 题意可能会很绕 一句话:基环树的直径. 求直径: 对于环上每一个点记录其向它的子树最长路径为$dp_ ...
- BZOJ 2806: [Ctsc2012]Cheat [广义后缀自动机 单调队列优化DP 二分]
2806: [Ctsc2012]Cheat 题意: 多个主串和多个询问串,每次询问将询问串分成多个连续子串,如果一个子串长度>=L且在主串中出现过就是熟悉的 如果熟悉的字符串长度>=询问串 ...
- bzoj 2806 [Ctsc2012]Cheat——广义后缀自动机+单调队列优化DP
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2806 只想着怎么用后缀数据结构做,其实应该考虑结合其他算法. 可以二分那个长度 L .设当前 ...
- BZOJ 2442 [Usaco2011 Open]修剪草坪:单调队列优化dp
题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=2442 题意: 有n个数a[i]从左到右排成一排. 你可以任意选数,但是连续的数不能超过k个 ...
- bzoj 1791: [Ioi2008]Island 岛屿
#include<iostream> #include<cstdio> #define M 1000009 using namespace std; *M],cnt,n,hea ...
- BZOJ1791[Ioi2008]Island 岛屿 ——基环森林直径和+单调队列优化DP+树形DP
题目描述 你将要游览一个有N个岛屿的公园.从每一个岛i出发,只建造一座桥.桥的长度以Li表示.公园内总共有N座桥.尽管每座桥由一个岛连到另一个岛,但每座桥均可以双向行走.同时,每一对这样的岛屿,都有一 ...
随机推荐
- 笔记——python风格规范
分号 不要在行尾加分号, 也不要用分号将两条命令放在同一行. 行长度 每行不超过80个字符 例外: 长的导入模块语句 注释里的URL 不要使用反斜杠连接行. Python会将 圆括号, 中括号和花括号 ...
- JAVA IO中read()方法的返回值
read()方法的作用是从输入流读取数据的下一个字节,返回的字节的值是一个0~255之间的整数.到达流的末尾返回-1. 刚开始我以为这个返回值表示的是所读取的数据的长度,可是之后在一个示例程序中发现这 ...
- HDU1166 线段树裸题 区间求和
敌兵布阵 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total Submis ...
- hdu - 2822 Dogs (优先队列+bfs)
http://acm.hdu.edu.cn/showproblem.php?pid=2822 给定起点和终点,问从起点到终点需要挖几次只有从# 到 .或者从. 到 . 才需要挖一次. #includ ...
- mysql-performance-schema
http://www.fromdual.com/mysql-performance-schema-hints http://www.cnblogs.com/cchust/
- 纤程(FIBER)
Indy 10 还包含对纤程的支持.纤程是什么?简单来说,它也是 一个“线程”,但是它是由代码控制的,而不是由操作系统控制的.实际上,可以认为线程 是一个高级纤程.纤程和 Unix 用户线程(Unix ...
- android账号与同步之账号管理
在android提供的sdk中,samples文件夹下有一个叫SampleSyncAdapter的演示样例,它是一个账号与同步的实例,比方Google原始的android手机能够使用Google账号进 ...
- script标签async和defer的区别及作用
作用: 1.没有 defer 或 async,浏览器会立即加载并执行指定的脚本,也就是说不等待后续载入的文档元素,读到就加载并执行. 2.async 属性表示异步执行引入的 JavaScript,与 ...
- 关于MacBook怎么更新Android SDK
昨天公司的人给了我一个VPN,可是还是无法更新SDK,后来发现将下图: 通过VPN发送全部流量勾选以后就能够连接更新了,哎.处处皆学问,特此分享一下此经验. 喜欢的朋友关注我哦! 多谢支持
- Cocos2d-x3.1下 Android,APK自己主动升级
项目要做Android的自己主动升级,对于我们之前做iOS的转Cocos开发做Android方面的功能..... 不正确说了.这里记录下我的实现过程. 原文地址:http://blog.csdn.ne ...