题目描述

你将要游览一个有N个岛屿的公园。从每一个岛i出发,只建造一座桥。桥的长度以Li表示。公园内总共有N座桥。尽管每座桥由一个岛连到另一个岛,但每座桥均可以双向行走。同时,每一对这样的岛屿,都有一艘专用的往来两岛之间的渡船。  相对于乘船而言,你更喜欢步行。你希望所经过的桥的总长度尽可能的长,但受到以下的限制。  • 可以自行挑选一个岛开始游览。  • 任何一个岛都不能游览一次以上。  • 无论任何时间你都可以由你现在所在的岛S去另一个你从未到过的岛D。由S到D可以有以下方法:  o 步行:仅当两个岛之间有一座桥时才有可能。对于这种情况,桥的长度会累加到你步行的总距离;或者  o 渡船:你可以选择这种方法,仅当没有任何桥和/或以前使用过的渡船的组合可以由S走到D(当检查是否可到达时,你应该考虑所有的路径,包括经过你曾游览过的那些岛)。  注意,你不必游览所有的岛,也可能无法走完所有的桥。  任务  编写一个程序,给定N座桥以及它们的长度,按照上述的规则,计算你可以走过的桥的最大长度。  限制  2 <= N <= 1,000,000 公园内的岛屿数目。  1<= Li <= 100,000,000 桥i的长度。 

输入

• 第一行包含N个整数,即公园内岛屿的数目。岛屿由1到N编号。  • 随后的N行每一行用来表示一个岛。第i 行由两个以单空格分隔的整数,表示由岛i筑的桥。第一个整数表示桥另一端的岛,第二个整数表示该桥的长度Li。你可以假设对於每座桥,其端点总是位于不同的岛上。 

输出

你的程序必须向标准输出写出包含一个整数的单一行,即可能的最大步行距离。  注1:对某些测试,答案可能无法放进32-bit整数,你要取得这道题的满分,可能需要用Pascal的int64或C/C++的long long类型。  注2:在比赛环境运行Pascal程序,由标准输入读入64-bit数据比32-bit数据要慢得多,即使被读取的数据可以32-bit表示。我们建议把输入数据读入到32-bit数据类型。  评分  N不会超过4,000。 

样例输入

7
3 8
7 2
4 2
1 4
1 9
3 4
2 3

样例输出

24

提示

  题意就是求基环森林中的每棵基环树的直径之和,重点是求基环树直径。基环树可以看做是一个环,环上的一些点向外连了一棵树(我们可以称这些基环树里的小树为外向树)。

  而基环树的直径有两种可能:1、直径的两端都在同一棵外向树中,即基环树的直径就是这棵外向树的直径。2、直径的两端分别在两棵外向树中,直径绕过环的一部分(也有可能端点在环上,可以看做这个端点在只有一个点的外向树的根节点处)。

  对于第一种情况直接每棵外向树求直径就好了,但有几点要注意:

1、求直径方法应该用从根节点找到最远点再从最远点找到它的最远点,而不是维护每个点向下的最长链和与最长链不重合的次长链——因为第二种方法递归时判断的东西太多会爆栈。

2、第一次dfs不要把遍历的点标记为已访问过(即used[x]=-1),要在第二次dfs再标记。在第一次dfs之前要把环上那个点标记置0,否则第二次dfs时遍历不到那个点。

  对于第二种情况将环从一个基准点拆开后倍长,计算出每个点到基准点的距离,再把每个点向外向树延伸的最深距离作为这个点的点权。在这个倍长的序列上DP,因为数据范围较大,所以要用单调队列优化DP。假设序列上的点分别是A1,B1,C1,D1,A2,B2,C2,D2,A到B在环上的顺时针距离就是B1-A1,逆时针距离就是A2-B1。当以D1作为直径的右端点时,A1——D1的直径长是A1的点权+D1的点权+A1B1+B1C1+C1D1,而B1——D1的直径长是B1的点权+D1的点权+B1C1+C1D1。我们发现只要B1的点权>A1的点权+A1B1,选B1作为左端点就比A1更优。只要枚举右端点,用单调队列扫一遍就OK了。

最后附上代码。

#include<set>
#include<map>
#include<queue>
#include<cmath>
#include<stack>
#include<vector>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
typedef long long ll;
using namespace std;
int n;
int cnt;
int tot;
int x;
ll y;
ll ans;
ll f[1000010];
int g[1000010];
int head[1000010];
int to[2000010];
int next[2000010];
int val[2000010];
int used[1000010];
int vis[1000010];
int q[2000010];
int a[1000010];
int pre[1000010];
int suf[1000010];
ll num[2000010];
int cost[1000010];
ll s[2000010];
ll mx;
ll answer;
ll S;
void add(int x,int y,int z)
{
tot++;
next[tot]=head[x];
head[x]=tot;
to[tot]=y;
val[tot]=z;
}
void find(int x)
{
vis[x]=1;
for(int i;;x=i)
{
i=suf[x];
if(vis[i])
{
a[0]=i;
s[1]=cost[x];
used[i]=-1;
for(int j=x;j!=i;j=pre[j])
{
a[++cnt]=j;
s[cnt+1]=s[cnt]+cost[pre[j]];
used[j]=-1;
}
cnt++;
return ;
}
pre[i]=x;
vis[i]=1;
}
}
void tree_dp(int x,int fa)
{
g[x]=x;
f[x]=0;
for(int i=head[x];i;i=next[i])
{
if(to[i]!=fa&&used[to[i]]!=-1)
{
tree_dp(to[i],x);
if(f[to[i]]+val[i]>f[x])
{
f[x]=f[to[i]]+val[i];
g[x]=g[to[i]];
}
}
}
}
void tree_dp2(int x,int fa)
{
g[x]=x;
f[x]=0;
used[x]=-1;
for(int i=head[x];i;i=next[i])
{
if(to[i]!=fa&&used[to[i]]!=-1)
{
tree_dp2(to[i],x);
if(f[to[i]]+val[i]>f[x])
{
f[x]=f[to[i]]+val[i];
g[x]=g[to[i]];
}
}
}
}
ll dis(int x,int y)
{
return s[y-1]-s[x-1];
}
ll queue_dp()
{
ll res=0;
for(int i=1;i<=cnt;i++)
{
num[i+cnt]=num[i];
}
for(int i=cnt+1;i<=2*cnt;i++)
{
s[i]=s[i-cnt]+S;
}
int l=1;
int r=0;
for(int i=1;i<=2*cnt;i++)
{
while(l<=r&&i-q[l]>=cnt)
{
l++;
}
if(l<=r)
{
res=max(res,num[i]+num[q[l]]+dis(q[l],i));
}
while(l<=r&&num[i]>=num[q[r]]+dis(q[r],i))
{
r--;
}
q[++r]=i;
}
return res;
}
int main()
{
scanf("%d",&n);
for(int i=1;i<=n;i++)
{
scanf("%d%d",&x,&y);
add(i,x,y);
add(x,i,y);
suf[i]=x;
cost[i]=y;
}
for(int i=1;i<=n;i++)
{
if(used[i]==0)
{
cnt=0;
find(i);
mx=0;
S=s[cnt];
for(int j=0;j<cnt;j++)
{
used[a[j]]=0;
tree_dp(a[j],0);
answer=g[a[j]];
num[j+1]=f[a[j]];
tree_dp2(answer,0);
mx=max(mx,f[answer]);
}
mx=max(queue_dp(),mx);
ans+=mx;
}
}
printf("%lld",ans);
}

BZOJ1791[Ioi2008]Island 岛屿 ——基环森林直径和+单调队列优化DP+树形DP的更多相关文章

  1. [bzoj1791][ioi2008]Island 岛屿(基环树、树的直径)

    [bzoj1791][ioi2008]Island 岛屿(基环树.树的直径) bzoj luogu 题意可能会很绕 一句话:基环树的直径. 求直径: 对于环上每一个点记录其向它的子树最长路径为$dp_ ...

  2. BZOJ1791 [Ioi2008]Island 岛屿[基环树+单调队列优化DP]

    基环树直径裸题. 首先基环树直径只可能有两种形式:每棵基环树中的环上挂着的树的直径,或者是挂在环上的两个树的最大深度根之间的距离之和. 所以,先对每个连通块跑一遍,把环上的点找出来,然后对环上每个点跑 ...

  3. BZOJ1791: [Ioi2008]Island 岛屿

    BZOJ1791: [Ioi2008]Island 岛屿 Description 你将要游览一个有N个岛屿的公园. 从每一个岛i出发,只建造一座桥. 桥的长度以Li表示. 公园内总共有N座桥. 尽管每 ...

  4. bzoj1791: [Ioi2008]Island 岛屿 单调队列优化dp

    1791: [Ioi2008]Island 岛屿 Time Limit: 20 Sec  Memory Limit: 162 MBSubmit: 1826  Solved: 405[Submit][S ...

  5. bzoj千题计划114:bzoj1791: [Ioi2008]Island 岛屿

    http://www.lydsy.com/JudgeOnline/problem.php?id=1791 就是求所有基环树的直径之和 加手工栈 #include<cstdio> #incl ...

  6. bzoj1791[IOI2008]Island岛屿(基环树+DP)

    题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=1791 题目大意:给你一棵n条边的基环树森林,要你求出所有基环树/树的直径之和.n< ...

  7. BZOJ 1791: [IOI2008]Island 岛屿 - 基环树

    传送门 题解 题意 = 找出无向基环树森林的每颗基环树的直径. 我们首先需要找到每颗基环树的环, 但是因为是无向图,用tarjan找环, 加个手工栈, 我也是看了dalao的博客才知道tarjan找无 ...

  8. [BZOJ1791][IOI2008]Island岛屿(环套树DP)

    同NOI2013快餐店(NOI出原题?),下面代码由于BZOJ栈空间过小会RE. 大致是对每个连通块找到环,在所有内向树做一遍DP,再在环上做两遍前缀和优化的DP. #include<cstdi ...

  9. bzoj 1791: [Ioi2008]Island 岛屿【基环树+单调队列优化dp】

    我太菜了居然调了一上午-- 这个题就是要求基环树森林的基环树直径和 大概步骤就是找环->dp找每个环点最远能到达距离作为点权->复制一倍环,单调队列dp 找环是可以拓扑的,但是利用性质有更 ...

随机推荐

  1. python简介及安装配置

    概述 python是解释型语言,相对编译型语言,执行效率较低.python是通过c语言编写,官方解释器也是c语言编写cpython,也有其他的如用java编写的jpython.目前有2.0和3.0版本 ...

  2. 浅谈Java泛型中的extends和super关键字

    泛型是在Java 1.5中被加入了,这里不讨论泛型的细节问题,这个在Thinking in Java第四版中讲的非常清楚,这里要讲的是super和extends关键字,以及在使用这两个关键字的时候为什 ...

  3. C# Hashtable vs Dictionary 学习笔记

    Hashtable 和 Dictionary 存储的都是键值对,我的理解是Dictionary是Hashtable的泛型实现. Hashtable的键和值都是object类型.所以,key和value ...

  4. sqlserver 发送http请求

    sp_configure 'show advanced options', 1; GO RECONFIGURE; GO sp_configure 'Ole Automation Procedures' ...

  5. Ubuntu16.04下完美切换Python版本

    http://blog.csdn.net/beijiu5854/article/details/77897767

  6. Linux mount 命令

    mount 命令用来挂载文件系统.其基本命令格式为:mount -t type [-o options] device dirdevice:指定要挂载的设备,比如磁盘.光驱等.dir:指定把文件系统挂 ...

  7. Docker容器学习梳理 - 日常操作总结

    使用Docker已有一段时间了,今天正好有空梳理下自己平时操作Docker时的一些命令和注意细节: Docker 命令帮助 $ sudo docker Commands: attach Attach ...

  8. HTTP协议基础与web服务的重定向,跳转以及请求转发

    JavaWeb中,HttpServletRequest与HttpServletResponse几乎是处理各种请求与操作必备的参数,与原始的ServletRequest/ServletResponse相 ...

  9. [LeetCode] 307. Range Sum Query - Mutable 解题思路

    Given an integer array nums, find the sum of the elements between indices i and j (i ≤ j), inclusive ...

  10. 数据处理项目Postmortem

    数据处理项目Postmortem 1. 设想和目标 1)目标我们的软件要解决什么问题?是否定义得很清楚?是否对典型用户和典型场景有清晰的描述? 我们的项目是学霸系统PipeLine,软件主要解决学霸系 ...