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) ...
随机推荐
- PHP、Java、Python、C、C++ 这几种编程语言都各有什么特点或优点
PHP.Java.Python.C.C++ 这几种编程语言都各有什么特点或优点 汇编: C: Java: C#: PHP: Python: Go: Haskell: Lisp: C++: &l ...
- session 与 cookie (一)
服务器信息临时存储 session篇 web.xml设置 <session-config> <session-timeout></session-timeout> ...
- 原始的Ajax方法 (异步的 JavaScript 和 XML -- (Extensible Markup Language 可扩展标记语言))
<script language="javascript" type="text/javascript"> var request = false; ...
- 进军ABP第一天:ABP理论知识
1.2.3 领域层领域层就是业务层,是一个项目的核心,所有业务规则都应该在领域层实现. ( 实体(Entity ) 实体代表业务领域的数据和操作,在实践中,通过用来映射成数据库表. ( 仓储(Repo ...
- MongoDb进阶实践之三 MongoDB查询命令详述
一.引言 上一篇文章我们已经介绍了MongoDB数据库的最基本操作,包括数据库的创建.使用和删除数据库,文档的操作也涉及到了文档的创建.删除.更新和查询,当然也包括集合的创建.重命 ...
- Linux后台运行命令 nohup command > myout.file 2>&1
Linux命令后台运行 转自北国的雨,谢谢:http://www.cnblogs.com/lwm-1988/archive/2011/08/20/2147299.html 有两种方式:1. comma ...
- 使用java 打印日历
package hangshu; /* * 打印从1900年到2.year年的日历 */ import java.util.Scanner; public class Calender { publi ...
- 【Java】0X003 面向对象
一. 什么是面向对象 都说Java是一门面向对象的语言,但什么对象?什么又是面向对象?以下都是我学到的知识和一点自己的理解. 对象是指包含属性和行为的主体. 比如,人有性别.血型.单眼皮或双眼皮等的特 ...
- 浅谈 DML、DDL、DCL的区别
一.DML DML(data manipulation language)数据操纵语言: 就是我们最经常用到的 SELECT.UPDATE.INSERT.DELETE. 主要用来对数据库的数据进行一些 ...
- C# 6.0中你不知道的新特性
为什么写? 今天去上班的公交上,有朋友在张队(张善友)的微信群里,发了一个介绍C# 6.0新特性的视频,视频7分钟,加上本人英语实在太low,整体看下来是一脸懵逼的. 下班回到家里,打开这个视频,把视 ...