NOIP 2013 【货车运输】

NOIP 2013 【货车运输】_第1张图片


【题目大意】给定一张无向图 以及若干个询问 对于每个询问求所有由节点u到节点v的路径上边权的最小值的最大值。

【题解】

  首先用构造一棵最大生成树,这样保证树上两个节点路径边权的最小值最大

  在最大生成树上两个节点之间只有一条路径,所以只需要找路径上边权的最小值

 为了快速的寻找最小值,利用树上倍增的想法用f[j][i]记录j的第2^i个祖先 并用 g[j][i]记录j到f[j][i]的路径上边权的最小值

 然后在找公共祖先时顺便就能处理最小值

【注意】可能不是一张连通图,所有在lca时记得判断两个点是不是在同一棵树里

 详见代码

 

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
using namespace std;
int i,j,k,l,m,n,fx,fy,q,x,y,z,num;
int fa[10005],up[10005][20],first[10005],g[10005][20],h[10005];
struct info
  {
  	int fr,ar,l,next;
  }road[100050],tree[20005];
//vector  tree[10005];
bool cmp(info x,info y) {return x.l>y.l;}
int getfa(int x)
  {
  	if(fa[x]==x) return x;else return fa[x]=getfa(fa[x]);
  }
void dfs(int u)
  {
  	int i,v;
  	for (i=first[u];i;i=tree[i].next)
	   {
	   	 v=tree[i].ar;
	   	 if (!h[v])
	   	   {
	   	   	 h[v]=h[u]+1;
	   	   	 up[v][0]=u;g[v][0]=tree[i].l;
	   	   	 dfs(v);
		   }
	   } 
  }
int lca(int x,int y)
  {
  	if (getfa(x)!=getfa(y)) return -1;
  	int i,j,ans=1e9;
  	if (h[y]>h[x]) swap(x,y);
  	for (;h[x]>h[y];)
  	  {
  	  	for (i=0;h[up[x][i]]>=h[y];i++);i--;
  	  	ans=min(ans,g[x][i]);
  	  	x=up[x][i];
	  }
	for (;x!=y;)
	  {
	  	for (i=1;up[x][i]!=up[y][i];i++);i--;
	  	ans=min(ans,min(g[x][i],g[y][i]));
	  	x=up[x][i];y=up[y][i];
	  }
	return ans;
  }
int main() 
  {
  	scanf("%d%d",&n,&m);
  	for (i=1;i<=m;i++)
  	  scanf("%d%d%d",&x,&y,&z),road[i]=(info){x,y,z,0};
  	sort(road+1,road+m+1,cmp);
    for (i=1;i<=n;i++) fa[i]=i;
  	for (i=1;i<=m;i++)
  	  {
  	  	fx=getfa(road[i].fr);fy=getfa(road[i].ar);
  	  	if (fx!=fy)
  	  	  {
  	  	  	fa[fx]=fy;
			tree[++num]=(info){road[i].fr,road[i].ar,road[i].l,first[road[i].fr]};first[road[i].fr]=num;
			tree[++num]=(info){road[i].ar,road[i].fr,road[i].l,first[road[i].ar]};first[road[i].ar]=num;
		  }
	  }
	for (i=1;i<=n;i++)
	  if (!h[i])
	    {
	      h[i]=1;
	      dfs(1);
		  up[i][0]=i;g[i][0]=1e9; 
		}
	for (i=1;i<=15;i++) 
	  for (j=1;j<=n;j++) 
	    {
	      up[j][i]=up[up[j][i-1]][i-1];
		  g[j][i]=min(g[j][i-1],g[up[j][i-1]][i-1]);
		} 
    for (scanf("%d",&q);q;q--)
      {
      	scanf("%d%d",&x,&y);
      	printf("%d\n",lca(x,y));
	  }
  }

 

你可能感兴趣的:(lca,倍增,图论)