[NOIP2013]货车运输

【题目描述】

思路{直接最大生成树+树链剖分即可。}

 1 #include<iostream>
 2 #include<cstdio>
 3 #include<cstring>
 4 #include<algorithm>
 5 #include<cmath>
 6 #include<queue>
 7 #include<vector>
 8 #include<set>
 9 #include<map>
10 #define LL long long 
11 #define dd double 
12 #define Inf 0x3f3f3f3f
13 #define maxx 10001
14 #define maxm 50001
15 #define rs ((o<<1)|1)
16 #define ls (o<<1)
17 #define mid ((l+r)>>1)
18 #define I(i) que[i]
19 #define RG register
20 using namespace std;
21 struct ee{int nxt,to,c;}e[maxm*2],E[maxm*2];int siz,W[maxx],w[maxx],hson[maxx];
22 int head[maxx],n,m,fa[maxx],col[maxx],HEAD[maxx],tot,TOT,colorr,f[maxx],top[maxx],sz[maxx],deep[maxx],id[maxx],cn,tree[maxx*4];
23 int find(int x){if(fa[x]!=x)fa[x]=find(fa[x]);return fa[x];}
24 int que[maxm*2];struct bi{int u,v,c;}b[maxm];vector<int>ha[maxx];
25 void ADD(int u,int v,int c){E[TOT].nxt=HEAD[u];E[TOT].to=v;E[TOT].c=c;HEAD[u]=TOT++;}
26 void add(int u,int v,int c){e[tot].nxt=head[u];e[tot].to=v;e[tot].c=c;head[u]=tot++;}
27 inline int read(){
28   int res=0,fh=1;char ch=getchar();
29   while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
30   if(ch=='-')fh=-1,ch=getchar();
31   while(ch<='9'&&ch>='0')res=res*10+ch-'0',ch=getchar();
32   return res*fh;
33 }
34 void DFS(int u,int faa){
35   for(int i=0;i<ha[u].size();++i)que[++que[0]]=ha[u][i];
36   for(int i=HEAD[u];i!=-1;i=E[i].nxt)if(!col[E[i].to]){
37       siz++;int v=E[i].to;col[v]=col[u];DFS(v,u);
38     }
39 }
40 bool comp(const int & aa,const int & bb){return b[aa].c>b[bb].c;}
41 void dfs(int u,int faa){
42   f[u]=faa;sz[u]=1;deep[u]=deep[faa]+1;
43   for(int i=head[u];i!=-1;i=e[i].nxt)if(!sz[e[i].to]){
44       int v=e[i].to;W[v]=e[i].c;dfs(v,u);
45       sz[u]+=sz[v];if(sz[hson[u]]<sz[v])hson[u]=v;
46     }
47 }
48 void dfs2(int u,int toop){top[u]=toop;
49   id[u]=++cn;w[cn]=W[u];if(hson[u])dfs2(hson[u],toop);
50   for(int i=head[u];i!=-1;i=e[i].nxt)if(e[i].to!=f[u]&&e[i].to!=hson[u])dfs2(e[i].to,e[i].to);
51 }
52 void kurusal(int u){siz=1;memset(que,0,sizeof(que));
53   col[u]=++colorr;DFS(u,u);sort(que+1,que+que[0]+1,comp);int cnt=0;
54   for(int i=1;i<=que[0]&&cnt+1!=siz;++i)
55     if(fa[find(b[I(i)].u)]!=fa[find(b[que[i]].v)])
56       fa[find(b[I(i)].u)]=fa[find(b[I(i)].v)],
57       add(b[I(i)].u,b[I(i)].v,b[I(i)].c),add(b[I(i)].v,b[I(i)].u,b[I(i)].c);
58   dfs(u,u);dfs2(u,u);
59 }
60 void build(int o,int l,int r){
61   if(l==r){tree[o]=w[l];return;}
62   build(ls,l,mid),build(rs,mid+1,r);
63   tree[o]=min(tree[rs],tree[ls]);
64 }
65 int query(int o,int l,int r,int L,int R){
66   if(L<=l&&r<=R)return tree[o];
67   if(mid<L)return query(rs,mid+1,r,L,R);
68   else if(mid>=R)return query(ls,l,mid,L,R);
69   else return min(query(rs,mid+1,r,L,R),query(ls,l,mid,L,R));
70 }
71 void lca(int x,int y){
72   int ans=Inf;
73   while(top[x]!=top[y]){
74     if(deep[top[x]]<deep[top[y]])swap(x,y);
75     ans=min(ans,query(1,1,n,id[top[x]],id[x]));
76     x=f[top[x]];
77   }
78   if(deep[x]<deep[y])swap(x,y);
79   if(id[y]<id[x])ans=min(ans,query(1,1,n,id[y]+1,id[x]));
80   printf("%d\n",ans);
81 }
82 int main(){
83   memset(HEAD,-1,sizeof(HEAD));
84   memset(head,-1,sizeof(head));scanf("%d%d",&n,&m);
85   for(RG int i=1;i<=n;++i)fa[i]=i;
86   for(int i=1;i<=m;++i)b[i].u=read(),b[i].v=read(),b[i].c=read(),ha[b[i].u].push_back(i),ADD(b[i].u,b[i].v,b[i].c),ADD(b[i].v,b[i].u,b[i].c);
87   for(int i=1;i<=n;++i)if(!col[i])kurusal(i);int kk;scanf("%d",&kk);build(1,1,n);
88   for(int i=1;i<=kk;++i){
89     int x,y;scanf("%d%d",&x,&y);
90     if(col[x]!=col[y]){cout<<"-1\n";continue;}
91    else lca(x,y);
92   }
93   return 0;
94 }

 

posted @ 2017-05-27 23:58  QYP_2002  阅读(177)  评论(0编辑  收藏  举报