[bzoj2125]最短路

传送门

仙人掌最短路,圆方树!
先dfs找环,顺带求出环的长度len,记录每个点到环内深度最小的点的最短路径有没有经过返祖边
对于每个环建一个方点,都是圆方树的基本操作啦!
求答案时对于\(dist(x,y)\),先求出\(z=lca(x,y)\),然后判断\(z\)是否是方点
1、z是方点,那么x和y的最短路径经过了一个环,那么我们对于x和y就少跳一步,跳到z的儿子处。
对应的圆方树是这样的:

原图上是这样的:

很显然这个时候需要分类讨论了:(x,y是跳了后的x,y)
假如x和y的最短路径都经过返祖边或者都不经过返祖边,并且不知道t-x和t-y哪条是返祖边的情况下,最短距离就是\(min(abs(dep[x]-dep[y]),min(len-dep[x]+dep[y],len-dep[y]+dep[x]))\)
否则就是只有一条经过返祖边,答案就是\(min(len-(dep[x]-dep[z])-(dep[y]-dep[z]),(dep[x]-dep[z])+(dep[y]-dep[z]))\)

2、z不是方点,那么直接算就是了,答案就是\(dep[x]+dep[y]-2*dep[z]\)(x,y是读入的x,y)

代码:

#include<cstdio>
#include<iostream>
#include<algorithm>
#include<cstring>
using namespace std;
void read(int &x) {
	char ch; bool ok;
	for(ok=0,ch=getchar(); !isdigit(ch); ch=getchar()) if(ch=='-') ok=1;
	for(x=0; isdigit(ch); x=x*10+ch-'0',ch=getchar()); if(ok) x=-x;
}
#define rg register
const int maxn=2e4+1e3;
int n,m,q,dep[maxn],top,id,f[maxn][21],dis[maxn],len[maxn],ans;bool vis[maxn*2],w[maxn*2];
struct o{int x,id;}st[maxn];
struct oo{
	int pre[maxn*2],nxt[maxn*2],h[maxn*2],v[maxn*2],cnt;
	oo(){cnt=1;}
	inline void add(int x,int y,int z)
	{
		pre[++cnt]=y,nxt[cnt]=h[x],h[x]=cnt,v[cnt]=z;
		pre[++cnt]=x,nxt[cnt]=h[y],h[y]=cnt,v[cnt]=z;
	}
}a,b;
void dfs(int x,int fa)
{
	for(rg int i=a.h[x];i;i=a.nxt[i])
		if(a.pre[i]!=fa)
		{
			if(!dep[a.pre[i]])st[++top].x=x,st[top].id=i,dep[a.pre[i]]=dep[x]+a.v[i],dfs(a.pre[i],x),top--;
			else if(dep[a.pre[i]]<dep[x])
			{
				++id,b.add(id,x,min(dep[x]-dep[a.pre[i]],a.v[i]));
				int g=min(dep[x]-dep[a.pre[i]],a.v[i]);len[id-n]+=a.v[i];
				if(g==a.v[i])w[x]=1;vis[i]=vis[i^1]=1;int ttt=top;
				while(a.pre[i]!=st[top].x)
				{
					int t=st[top].x,g=min(dep[t]-dep[a.pre[i]],dep[x]-dep[t]+a.v[i]);vis[st[top].id]=vis[st[top].id^1]=1;
					if(g==dep[x]-dep[t]+a.v[i])w[t]=1;b.add(id,t,g),len[id-n]+=a.v[st[top].id],top--;
				}
				int t=st[top].x;b.add(id,t,0);len[id-n]+=a.v[st[top].id];
				vis[st[top].id]=vis[st[top].id^1]=1;top=ttt;
			}
		}
}
void dfs1(int x,int fa)
{
	for(rg int i=1;i<=20;i++)
	{
		if(dis[x]<(1<<i))break;
		f[x][i]=f[f[x][i-1]][i-1];
	}
	for(rg int i=b.h[x];i;i=b.nxt[i])
		if(b.pre[i]!=fa)f[b.pre[i]][0]=x,dis[b.pre[i]]=dis[x]+1,dep[b.pre[i]]=dep[x]+b.v[i],dfs1(b.pre[i],x);
}
int lca(int x,int y)
{
	if(dis[x]>dis[y])swap(x,y);
	int poor=dis[y]-dis[x];
	for(rg int i=20;i>=0;i--)if(poor&(1<<i))y=f[y][i];
	for(rg int i=20;i>=0;i--)if(f[y][i]!=f[x][i])x=f[x][i],y=f[y][i];
	return x==y?x:f[x][0];
}
int jump(int x,int z)
{
	int poor=dis[x]-dis[z]-1;
	for(rg int i=20;i>=0;i--)if(poor&(1<<i))x=f[x][i];
	return x;
}
int main()
{
	read(n),read(m),read(q);id=n;
	for(rg int i=1,x,y,z;i<=m;i++)read(x),read(y),read(z),a.add(x,y,z);
	dep[1]=1,dfs(1,0);for(int i=2;i<=a.cnt;i+=2)if(!vis[i])b.add(a.pre[i],a.pre[i^1],a.v[i]);
	memset(dep,0,sizeof dep),dfs1(1,0);
	for(rg int i=1,x,y,z,l,r;i<=q;i++)
	{
		read(x),read(y);z=lca(x,y);ans=0;
		if(z>n)
		{
			l=jump(x,z),r=jump(y,z);
			ans=dep[x]-dep[l]+dep[y]-dep[r];
			if((w[l]&&w[r])||(!w[l]&&!w[r]))ans+=min(abs(dep[l]-dep[r]),min(len[z-n]-dep[l]+dep[r],len[z-n]-dep[r]+dep[l]));
			else ans+=min(len[z-n]-(dep[l]-dep[z])-(dep[r]-dep[z]),(dep[l]-dep[z])+(dep[r]-dep[z]));
		}
		else ans=dep[x]+dep[y]-dep[z]*2;
		printf("%d\n",ans);
	}
}
posted @ 2019-01-17 17:54  蒟蒻--lichenxi  阅读(168)  评论(0编辑  收藏  举报