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) ...
随机推荐
- 51Nod P1100 斜率最大
传送门: https://www.51nod.com/onlineJudge/questionCode.html#!problemId=1100 由于2 <= N <= 10000, 所以 ...
- c#+wpf项目性能优化之OutOfMemoryException解密
近期,使用c#+wpf开发的软件准备正式投入使用了,使用前进行了大量的测试,测试后发现了一些问题,其中最让人头疼的就是软件的性能问题(稳定性). 这里的稳定性具体表现在机器的cpu占有率和内存使用情况 ...
- angular4学习笔记整理(二)angular4的路由使用
这章说一下angular的路由 先说angular路由怎么引入,一开始new出来的angular项目它路由帮你配好了,但看要看app.module.ts里面 1.首先最上面要引入路由模块 import ...
- List集合就这么简单【源码剖析】
前言 声明,本文用得是jdk1.8 前一篇已经讲了Collection的总览:Collection总览,介绍了一些基础知识. 现在这篇主要讲List集合的三个子类: ArrayList 底层数据结构是 ...
- iot前台开发环境:搭建 SpringBoot+angularJs2
参考网站 Angular 中文官网:https://angular.cn/ 参考代码:https://ng.ant.design/#/components/dropdown npm install ...
- mosquitto安装和测试
一.安装 1.windows安装 安装完毕,更新安装目录的dll文件 2.linux安装 编译保存用户数据到数据库的插件 安装 3.启动 mosquitto mosquitto mosquitto_p ...
- Spring Security 入门(1-5)Spring Security - 匿名认证
匿名认证 对于匿名访问的用户,Spring Security 支持为其建立一个匿名的 AnonymousAuthenticationToken 存放在 SecurityContextHolder 中, ...
- Spring Security 入门(1-3-2)Spring Security - http元素 - intercept-url配置
http元素下可以配置登录页面,也可以配置 url 拦截. 1.直接配置拦截url和对应的访问权限 <security:http use-expressions="false" ...
- 未能加载文件或程序集“ RevitAPIUI.dll”
revit二次开发中遇到的问题 RevitAPIUI.dll 只能 Native Library 中执行: 脱离了Native Library,API是跑不起来的 . 检查程序流程:登录,配置,启动r ...
- Java-NIO(三):直接缓冲区与非直接缓冲区
直接缓冲区与非直接缓冲区的概念: 1)非直接缓冲区:通过 static ByteBuffer allocate(int capacity) 创建的缓冲区,在JVM中内存中创建,在每次调用基础操作系统的 ...