zoukankan      html  css  js  c++  java
  • [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
    ",ans);
    	}
    }
    
  • 相关阅读:
    线索二叉树
    正则表达式之后向引用
    进步的阶梯
    树和二叉树
    java 执行 exe 文件
    Electron + Vue如何实现不同窗口之间的通信(项目总结 第一个)
    liunx 修改 ip 地址
    桌面快捷工具
    微信小程序长列表组件 recycle-view 修改,使其可以下拉刷新
    微信小程序 textarea 文本滚动不了的bug
  • 原文地址:https://www.cnblogs.com/lcxer/p/10283615.html
Copyright © 2011-2022 走看看