[BZOJ1791][IOI2008]岛屿Island(基环树的直径)

传送门


    先转换一下题目所给的图:
    给一个n个点n条边的图,不一定连通,但是不连通的地方可以用没有权值的虚边连接起来,既然可以任意连虚边那么我们就可以把原图中的一个个块分开处理。

    然后分析这是什么图,因为一定有n个点n条边所以假如两个点之间没有边那么某一个块中一定会多出一条边来满足数量有n条。所以原图中每个连通块的边数=点数,刚好与基环树的性质契合,所以原图=基环树森林。

    那么答案就是对于将每个基环树的直径加起来,我们分别求出每一个基环树的直径。那么,问题转换为,求一个基环树的直径。

    基环树的直径定义为:基环树中最长的简单路径的长度(简单路径指不重复经过任何点或边的路径)。 那么基环树的直径只可能有两种情况
1、不经过环(在环上的某一点的子树中)
2、经过了环(某一段在环上)

    那么对于一个基环树首先一次dfs把环搞出来。
    对于第一种情况,我们对于环上的每一个点x求出他的子树的直径 d s o n [ x ] d_{son}[x] dson[x],那么第一种情况下的直径 D 1 D_1 D1

D 1 = m a x { d s o n [ x ] } D_1=max\lbrace d_{son}[x] \rbrace D1=max{dson[x]}

    对于第二种情况,首先对于环上的一点 x x x,求出他的子树中最深的点和与根节点距离 d e p [ x ] dep[x] dep[x],然后第二种情况下的直径 D 2 D_2 D2

D 2 = m a x { d e p [ i ] + d e p [ j ] + d i s t ( i , j ) } ( i , j ∈ 环 , d i s t ( i , j ) 表 示 i , j 在 环 上 的 最 大 距 离 ) D_2=max\lbrace dep[i]+dep[j]+dist(i,j) \rbrace(i,j \in 环,dist(i,j)表示i,j在环上的最大距离) D2=max{dep[i]+dep[j]+dist(i,j)}(i,j,dist(i,j)i,j)

    如何求?首先环上的距离有两条,顺时针和逆时针,所以我们把环展开,然后复制一份接在后面,对于边权求一个前缀和,前缀和之差顺逆时针都可以求出,顺时针相当于是 s u m [ i ] − s u m [ j ] sum[i]-sum[j] sum[i]sum[j],逆时针相当于 l o o p l e n − ( s u m [ i ] − s u m [ j ] ) loop_{len}-(sum[i]-sum[j]) looplen(sum[i]sum[j])

    又因为 d i s t ( i , j ) = ( s u m [ j ] − s u m [ i ] ) dist(i,j)=(sum[j]-sum[i]) dist(i,j)=(sum[j]sum[i]),要求每个i找到最大的 m a x { d e p [ j ] − s u m [ j ] } max\lbrace dep[j]-sum[j]\rbrace max{dep[j]sum[j]} ,下标和 m a x { d e p [ j ] − s u m [ j ] } max\lbrace dep[j]-sum[j]\rbrace max{dep[j]sum[j]}都有单调性,那么用单调队列维护 m a x { d e p [ j ] − s u m [ j ] } max\lbrace dep[j]-sum[j]\rbrace max{dep[j]sum[j]}即可。

代码来自wisdom,有详细注释。


//BZOJ1791
#include<cstdio>
#include<iostream>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<queue>
using namespace std;
typedef long long ll;
const int N=1e6+10;
inline int read()
{
	int x=0,f=1; char ch=getchar();
	while(ch<'0' || ch>'9'){if(ch=='-')f=-1; ch=getchar();}
	while(ch>='0' && ch<='9'){x=x*10+ch-'0'; ch=getchar();}
	return x*f;
}
struct edge
{
	int x,y,c,next;
}a[N*2]; int len,last[N];
int n;
struct node
{
	int w,to;
}fa[N],loop[N];
void ins(int x,int y,int c)
{
	a[++len].x=x;a[len].y=y;a[len].c=c;
	a[len].next=last[x];last[x]=len;
}

int vis[N],id=0,cnt=0;
void get_loop(int x)//扣环 
{
	vis[x]=++id;
	for(int k=last[x];k;k=a[k].next)
	{
		int y=a[k].y;
		if(y==fa[x].to) continue;
		if(vis[y])//如果访问到一个点之前被访问过那么久形成了一个环 
		{
			if(vis[y]<vis[x]) continue;//按照一个方向获取环(vis天然形成了一个方向,模拟可得) 这样的loop得到的环是连续的 
			loop[++cnt].to=y;
			loop[cnt].w=a[k].c;
			for(;y!=x;y=fa[y].to)
				loop[++cnt]=fa[y];
		}
		else
		{
			fa[y].to=x; fa[y].w=a[k].c;
			get_loop(y);	
		}
	}
}
ll mx=0,sum[N*2],dep[N*2];
int v[N],now,pos;
void dfs(int x,int Fa,ll w)
{
	if(w>=mx)
	{
		mx=w;
		pos=x;
	}
	for(int k=last[x];k;k=a[k].next)
	{
		int y=a[k].y;
		if(y!=Fa && (!v[y] || y==now)) //不访问到环上的点 但是第二次的时候可以访问到now 
			dfs(y,x,w+a[k].c);
	}
}
ll f(int i){return dep[i]-sum[i];}
int list[N*2],head,tail;
int main()
{
	memset(last,0,sizeof(last)); len=0;
	n=read();
	for(int i=1;i<=n;i++)
	{
		int x=read(),w=read();
		ins(i,x,w); ins(x,i,w);
	}
	ll ans=0;
	sum[0]=dep[0]=0;
	for(int i=1;i<=n;i++)
	{
		if(!vis[i])//处理每棵基环树 
		{
			cnt=0; id=0; ll t=0;
			get_loop(i); //loop记录环上的点信息 to为点编号 w为点与环上另一个点之间的边的权值 
			for(int j=1;j<=cnt;j++) v[loop[j].to]=1;//标记环上的点 
			for(int j=1;j<=cnt;j++)//cnt环的长度 
			{
				mx=0; now=-1; //now当前从环上的哪个点下去  mx子树的直径   
				dfs(loop[j].to,0,0);
				dep[j]=mx;  //第一次dfs后找到最深的点 
				
				now=loop[j].to;
				dfs(pos,0,0);//对子树求一次直径(两次dfs) 
				t=max(t,mx);//第一种情况:基环树的直径在子树里 
			}
			
			for(int j=1;j<=cnt;j++) dep[j+cnt]=dep[j];
			head=1,tail=0; //第二种情况:基环树的直径经过了环 
			for(int j=1;j<=2*cnt;j++)
			{//答案 dist(i,j)+dep[j]+dep[i]  ( dist(i,j)=(sum[j]-sum[i]) )
				if(j<=cnt) sum[j]=sum[j-1]+loop[j].w;
				else sum[j]=sum[j-1]+loop[j-cnt].w;
				if(head<=tail) t=max(t, f(list[head])+sum[j]+dep[j]);
				//要求每个j找到最大的f(k) ,那么用单调队列维护f(i)=max{dep[i]-sum[i]}
				while(head<=tail && f(list[tail])<=f(j)) tail--;
				list[++tail]=j;
				while(head<=tail && list[head]<=j-cnt+1) head++;	
			}
			ans+=t;
		}
	}
	printf("%lld\n",ans);
	return 0;
}
  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
题目描述 有一个 $n$ 个点的棋盘,每个点上有一个数字 $a_i$,你需要从 $(1,1)$ 走到 $(n,n)$,每次只能往右或往下走,每个格子只能经过一次,路径上的数字和为 $S$。定义一个点 $(x,y)$ 的权值为 $a_x+a_y$,求所有满足条件的路径中,所有点的权值和的最小值。 输入格式 第一行一个整数 $n$。 接下来 $n$ 行,每行 $n$ 个整数,表示棋盘上每个点的数字。 输出格式 输出一个整数,表示所有满足条件的路径中,所有点的权值和的最小值。 数据范围 $1\leq n\leq 300$ 输入样例 3 1 2 3 4 5 6 7 8 9 输出样例 25 算法1 (形dp) $O(n^3)$ 我们可以先将所有点的权值求出来,然后将其看作是一个有权值的图,问题就转化为了在这个图中求从 $(1,1)$ 到 $(n,n)$ 的所有路径中,所有点的权值和的最小值。 我们可以使用形dp来解决这个问题,具体来说,我们可以将这个图看作是一棵,每个点的父节点是它的前驱或者后继,然后我们从根节点开始,依次向下遍历,对于每个节点,我们可以考虑它的两个儿子,如果它的两个儿子都被遍历过了,那么我们就可以计算出从它的左儿子到它的右儿子的路径中,所有点的权值和的最小值,然后再将这个值加上当前节点的权值,就可以得到从根节点到当前节点的路径中,所有点的权值和的最小值。 时间复杂度 形dp的时间复杂度是 $O(n^3)$。 C++ 代码 算法2 (动态规划) $O(n^3)$ 我们可以使用动态规划来解决这个问题,具体来说,我们可以定义 $f(i,j,s)$ 表示从 $(1,1)$ 到 $(i,j)$ 的所有路径中,所有点的权值和为 $s$ 的最小值,那么我们就可以得到如下的状态转移方程: $$ f(i,j,s)=\min\{f(i-1,j,s-a_{i,j}),f(i,j-1,s-a_{i,j})\} $$ 其中 $a_{i,j}$ 表示点 $(i,j)$ 的权值。 时间复杂度 动态规划的时间复杂度是 $O(n^3)$。 C++ 代码

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值