bzoj1791: [Ioi2008]Island 岛屿 单调队列优化dp
1791: [Ioi2008]Island 岛屿
Time Limit: 20 Sec Memory Limit: 162 MB
Submit: 1826 Solved: 405
[Submit][Status][Discuss]
Description
Input
Output
Sample Input
3 8
7 2
4 2
1 4
1 9
3 4
2 3
Sample Output
HINT

Source
题目理解一下就变成了求多棵基环外向树直径的和
对于任意一棵基环外向树,可以先对于环上每个点i做一个树形dp求出i向外延伸的最大距离
如果直径不经过环 那么在做树形dp的时候就可以找出来
如果经过环,那么在环上dp:
把每个点向外延伸的最大距离当成点权,问题即转化为求环上两点i,j 要求dis(i,j)+v[i]+v[j]最大
把无向环看作有向环,即把序列倍增一次补在后面,这样即可实现单方向转移
单调队列维护转移,在转移长度达到环长的时候head++
http://blog.csdn.net/vmurder/article/details/38940815
我写错了很多地方:
1.维护单调队列单调性时求两点距离错了
2.先没想到直径可以不经过环
3.数组爆掉
4.爆栈
前面3点在经过3h的调试之后更正了
第四点无法优化,因为bzoj好像不让自己扩栈?
爆栈这个问题,其实可以避免
我是dfs找环 再dfs环求dp (不炸成瓜皮才怪)
看了看网上更优的方法,可以先求每个联通块,然后对联通块topsort,topsort的时候顺便转移dp
感觉自己宛如一个zz,又忘记了topsort求环
#include<bits/stdc++.h>
#define N 1000005
#define ll long long
#pragma comment(linker, "/STACK:102400000,102400000")
using namespace std;
int n,tot,cnt,tp,fg,num,siz[N],s[N],q[N<<],hd[N];
int cir[N],bl[N],g[N<<],d[N<<],vis[N];
ll len,ans[N],f[N],dis[N<<],dp[N];
struct edge{int v,w,next;}e[N<<];
void adde(int u,int v,int w){
e[++tot].v=v;
e[tot].w=w;
e[tot].next=hd[u];
hd[u]=tot;
}
void dfs1(int u,int fa){
if(vis[u]&&!bl[u]){
cir[++cnt]=u;
int tmp=s[tp];
while(){
bl[tmp]=cnt;--tp;
++siz[cnt];
if(tmp==u)break;
tmp=s[tp];
}
return;
}
if(vis[u])return;
vis[u]=;s[++tp]=u;
int tmp=;
for(int i=hd[u];i;i=e[i].next){
int v=e[i].v;
if(v==fa&&!tmp){tmp=;continue;}
dfs1(v,u);
}
--tp;
}
void getdp(int u,int fa,int id){
ll m1=,m2=,res;
for(int i=hd[u];i;i=e[i].next){
int v=e[i].v;
if(v==fa||bl[v]==id)continue;
getdp(v,u,id);res=e[i].w+dp[v];
dp[u]=max(dp[u],res);
if(res>m1)m2=m1,m1=res;
else if(res>m2)m2=res;
}
ans[id]=max(ans[id],m1+m2);
}
inline ll getd(int i,int j){
return dis[j]-dis[i];
} void solve(int id){
ll &mx=ans[id];int L=siz[id],tid=num;
for(int i=;i<=num;++i)
g[++tid]=g[i],d[tid]=d[i],dis[tid]=dis[i];
int h=,t=;q[]=;mx=max(dp[g[]],mx);
for(int i=;i<tid;++i){
while(h<t&&i-q[h]>=L)h++;
if(i>num&&q[h]>num)break;
if(i>num)f[i]=len-dis[q[h]]+dp[g[q[h]]]+dp[g[i]]+dis[i];
else f[i]=dis[i]-dis[q[h]]+dp[g[q[h]]]+dp[g[i]];
mx=max(f[i],mx);if(i>num)continue;
while(h<=t&&dp[g[q[t]]]+getd(q[t],i)<=dp[g[i]])t--;q[++t]=i;
}
}
void dfs2(int u,int fa,int id){
if(u==cir[id]&&fa){
solve(id);fg=;
return;
}
g[++num]=u;getdp(u,,id);int tmp=;
for(int i=hd[u];i&&!fg;i=e[i].next){
int v=e[i].v;
if(bl[v]!=id)continue;
if(v==fa&&!tmp){tmp=;continue;}
if(v!=cir[id]){
dis[num+]=dis[num]+e[i].w;
d[num+]=e[i].w;
}
else d[]=e[i].w;
len+=e[i].w;dfs2(v,u,id);
}
}
int main(){
scanf("%d",&n);
for(register int i=;i<=n;++i){
int v,w;
scanf("%d%d",&v,&w);
adde(i,v,w);adde(v,i,w);
}
for(register int i=;i<=n;i++)
if(!vis[i])tp=,dfs1(i,);
for(register int i=;i<=cnt;i++){
len=;fg=;num=;dis[]=;
dfs2(cir[i],,i);
}
ll all=;
for(register int i=;i<=cnt;i++)all+=ans[i];
printf("%lld\n",all);
return ;
}
爆栈dfs
然后是网上一个人的代码 用的topsort
#include <iostream>
#include <algorithm>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <queue>
#define M 1000005
#define LL long long
using namespace std;
struct edge
{
int y,ne,v;
}e[M*];
int h[M],v[M],c[M],du[M],q[*M],n,m,tot,t;
LL f[M],d[M],a[*M],b[*M];
void Addedge(int x,int y,int v)
{
tot++;
e[tot].y=y;
e[tot].ne=h[x];
h[x]=tot;
e[tot].v=v;
du[x]++;
}
void dfs(int x,int k)
{
v[x]=,c[x]=k;
for (int i=h[x];i;i=e[i].ne)
{
int y=e[i].y;
if (v[y]) continue;
dfs(y,k);
}
}
void Topsort()
{
int l=,r=,y;
for (int i=;i<=n;i++)
if (du[i]==) q[++r]=i;
while (l<=r)
{
int x=q[l];
for (int i=h[x];i;i=e[i].ne)
if (du[y=e[i].y]>)
{
du[y]--;
d[c[x]]=max(d[c[x]],f[x]+f[y]+e[i].v);
f[y]=max(f[y],f[x]+e[i].v);
if (du[y]==) q[++r]=y;
}
l++;
}
}
void Dp(int t,int x)
{
int m=,i,y=x;
do
{
a[++m]=f[y],du[y]=;
for (i=h[y];i;i=e[i].ne)
if (du[e[i].y]>)
{
y=e[i].y;
b[m+]=b[m]+e[i].v;
break;
}
}while (i);
if (m==)
{
int l=;
for (int i=h[y];i;i=e[i].ne)
if (e[i].y==x) l=max(l,e[i].v);
d[t]=max(d[t],f[x]+f[y]+l);
return;
}
for (int i=h[y];i;i=e[i].ne)
if (e[i].y==x)
{
b[m+]=b[m]+e[i].v;
break;
}
for (int i=;i<=m;i++)
{
a[m+i]=a[i];
b[m+i]=b[m+]+b[i];
}
int l,r;
q[l=r=]=;
for (int i=;i<*m;i++)
{
while (l<=r&&i-q[l]>=m)
l++;
d[t]=max(d[t],a[i]+a[q[l]]+b[i]-b[q[l]]);
while (l<=r&&a[q[r]]+b[i]-b[q[r]]<=a[i])
r--;
q[++r]=i;
}
}
int main()
{
scanf("%d",&n);
for (int i=;i<=n;i++)
{
int x,y;
scanf("%d%d",&x,&y);
Addedge(x,i,y);
Addedge(i,x,y);
}
memset(v,,sizeof(v));
t=;
for (int i=;i<=n;i++)
if (!c[i]) dfs(i,++t);
Topsort();
LL ans=0LL;
memset(v,,sizeof(v));
for (int i=;i<=n;i++)
if (du[i]>&&!v[c[i]])
{
v[c[i]]=;
Dp(c[i],i);
ans+=d[c[i]];
}
cout<<ans<<endl;
return ;
}
topsort
bzoj1791: [Ioi2008]Island 岛屿 单调队列优化dp的更多相关文章
- bzoj1791[IOI2008]Island岛屿(基环树+DP)
题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=1791 题目大意:给你一棵n条边的基环树森林,要你求出所有基环树/树的直径之和.n< ...
- BZOJ1791[Ioi2008]Island 岛屿 ——基环森林直径和+单调队列优化DP+树形DP
题目描述 你将要游览一个有N个岛屿的公园.从每一个岛i出发,只建造一座桥.桥的长度以Li表示.公园内总共有N座桥.尽管每座桥由一个岛连到另一个岛,但每座桥均可以双向行走.同时,每一对这样的岛屿,都有一 ...
- P4381 [IOI2008]Island(基环树+单调队列优化dp)
P4381 [IOI2008]Island 题意:求图中所有基环树的直径和 我们对每棵基环树分别计算答案. 首先我们先bfs找环(dfs易爆栈) 蓝后我们处理直径 直径不在环上,就在环上某点的子树上 ...
- BZOJ1791: [Ioi2008]Island 岛屿
BZOJ1791: [Ioi2008]Island 岛屿 Description 你将要游览一个有N个岛屿的公园. 从每一个岛i出发,只建造一座桥. 桥的长度以Li表示. 公园内总共有N座桥. 尽管每 ...
- [bzoj1791][ioi2008]Island 岛屿(基环树、树的直径)
[bzoj1791][ioi2008]Island 岛屿(基环树.树的直径) bzoj luogu 题意可能会很绕 一句话:基环树的直径. 求直径: 对于环上每一个点记录其向它的子树最长路径为$dp_ ...
- 单调队列优化DP,多重背包
单调队列优化DP:http://www.cnblogs.com/ka200812/archive/2012/07/11/2585950.html 单调队列优化多重背包:http://blog.csdn ...
- bzoj1855: [Scoi2010]股票交易--单调队列优化DP
单调队列优化DP的模板题 不难列出DP方程: 对于买入的情况 由于dp[i][j]=max{dp[i-w-1][k]+k*Ap[i]-j*Ap[i]} AP[i]*j是固定的,在队列中维护dp[i-w ...
- hdu3401:单调队列优化dp
第一个单调队列优化dp 写了半天,最后初始化搞错了还一直wa.. 题目大意: 炒股,总共 t 天,每天可以买入na[i]股,卖出nb[i]股,价钱分别为pa[i]和pb[i],最大同时拥有p股 且一次 ...
- Parade(单调队列优化dp)
题目连接:http://acm.hdu.edu.cn/showproblem.php?pid=2490 Parade Time Limit: 4000/2000 MS (Java/Others) ...
随机推荐
- Scrum 冲刺 第三日
Scrum 冲刺 第三日 目录 要求 项目链接 燃尽图 问题 今日任务 明日计划 成员贡献量 要求 各个成员今日完成的任务(如果完成的任务为开发或测试任务,需给出对应的Github代码签入记录截图:如 ...
- 《高级软件测试》11.16.Jira使用说明的撰写和操作视频的录制
今日任务完成情况如下: 小王:完成了测试管理工具jira的使用手册中,基本情况介绍.下载安装部分的撰写工作:小高:参考官方手册,结合自己的实际使用体会,对jira的基本组成及其工作流程进行了介绍:小陈 ...
- SpringMVC源码情操陶冶#task-executor解析器
承接Spring源码情操陶冶-自定义节点的解析.线程池是jdk的一个很重要的概念,在很多的场景都会应用到,多用于处理多任务的并发处理,此处借由spring整合jdk的cocurrent包的方式来进行深 ...
- EasyUI中easyui-combobox的onchange事件。
html: <select id="cbox" class="easyui-combobox" name="dept" style=& ...
- JavaScript作用域那些事
作用域 (1).作用域也叫执行环境(execution context)是JavaScript中一个重要的概念.执行环境定义了变量或函数有权访问的其他数据,决定了它们各自的行为.在JavaScript ...
- 复习HTML+CSS(4)
n HTML颜色表示 网页中的颜色有三种表示方法 颜色单词:blue.green.red.yellow 10进制表示:rgb(255,0,0).rgb(0,255,0).rgb(0,0,255) 1 ...
- POJ-1860 Currency Exchange---Bellman-Ford判断正环
题目链接: https://vjudge.net/problem/POJ-1860 题目大意: 我们的城市有几个货币兑换点.让我们假设每一个点都只能兑换专门的两种货币.可以有几个点,专门从事相同货币兑 ...
- Java面试题—初级(3)
21.ArrayList和Vector的区别 这两个类都实现了List接口(List接口继承了Collection接口),他们都是有序集合,即存储在这两个集合中的元素的位置都是有顺序的,相当于一种动态 ...
- java 面向对象编程。。。。
经过一周的学习(java),总结了许多,对java的理解,java的类型,运用,安装,基础语法,变量,常量,表达式,语句 java从C语言中继承了大量语言特性.java面向对象编程的基本特征,包括继承 ...
- 解决MySQL在修改列时因为外键依赖出错的问题
因为 favorite_food 中的 person_id 对 person 表中的 person_id 有外键依赖关系,所以在执行 ALTER TABLE person MODIFY person_ ...