1. 程式人生 > >4144: [AMPPZ2014]Petrol (多源最短路+最小生成樹+啟發式合並)

4144: [AMPPZ2014]Petrol (多源最短路+最小生成樹+啟發式合並)

limit make 兩個 不用 gree set emp while include

4144: [AMPPZ2014]Petrol

Time Limit: 10 Sec Memory Limit: 256 MB
Submit: 752 Solved: 298
[Submit][Status][Discuss]

Description

給定一個n個點、m條邊的帶權無向圖,其中有s個點是加油站。 每輛車都有一個油量上限b,即每次行走距離不能超過b,但在加油站可以補滿。 q次詢問,每次給出x,y,b,表示出發點是x,終點是y,油量上限為b,且保證x點和y點都是加油站,請回答能否從x走到y。

Input

第一行包含三個正整數n,s,m(2<=s<=n<=200000,1<=m<=200000),表示點數、加油站數和邊數。 第二行包含s個互不相同的正整數c[1],c[2],...c[s](1<=c[i]<=n),表示每個加油站。 接下來m行,每行三個正整數u[i],v[i],d[i](1<=u[i],v[i]<=n,u[i]!=v[i],1<=d[i]<=10000),表示u[i]和v[i]之間有一條長度為d[i]的雙向邊。 接下來一行包含一個正整數q(1<=q<=200000),表示詢問數。 接下來q行,每行包含三個正整數x[i],y[i],b[i](1<=x[i],y[i]<=n,x[i]!=y[i],1<=b[i]<=2*10^9),表示一個詢問。

Output

輸出q行。第i行輸出第i個詢問的答案,如果可行,則輸出TAK,否則輸出NIE。

Sample Input

6 4 5
1 5 2 6
1 3 1
2 3 2
3 4 3
4 5 5
6 4 5
4
1 2 4
2 6 9
1 5 9
6 5 8

Sample Output

TAK
TAK
TAK
NIE

HINT

Source

鳴謝Claris上傳


樸素是先寫個多源最短路把加油站的最小生成樹所有可能邊弄出來,然後後面按部就班地建個最小生成樹,然後對樹上寫個主席樹(st倍增)找u到v上最大值,和b比較一下。

那我們在把所有可能邊弄出來以後停一下2333,我們連邊不要對這條邊倆端點連,而是對他們所在塊的堆頭節點連,這個建出來在查詢最大值上是等效的。然後我們啟發式地合並堆,這樣可以把堆高度控制在logn,就不用再寫個主席樹啥的那麻煩了。直接兩個端點往上一個一個跳,logn的不會有事的。

技術分享圖片
  1 #include<bits/stdc++.h>
  2 #define clr(x) memset(x,0,sizeof(x))
  3 #define clr_1(x) memset(x,-1,sizeof(x))
  4 #define INF 0x3f3f3f3f
  5 #define LL long long
  6 #define pb push_back
  7 #define mod 1000000007
  8 #define ls(i) (i<<1)
  9 #define rs(i) (i<<1|1)
 10 #define mp make_pair
 11
#define fi first 12 #define se second 13 using namespace std; 14 typedef pair<LL,int > pli; 15 const int N = 2e5+10; 16 bool need[N]; 17 vector<pli> e[N]; 18 struct node 19 { 20 int u,v; 21 LL w; 22 node(int _u=0,int _v=0,LL _w=0):u(_u),v(_v),w(_w) {} 23 }; 24 vector<node> ve; 25 int n,s,m,T; 26 priority_queue<pli,vector<pli>,greater<pli> > que; 27 int fuel[N]; 28 int fa[N],rfa[N],rk[N],dep[N]; 29 LL dis[N],val[N]; 30 bool vis[N]; 31 int pre[N]; 32 void dij() 33 { 34 while(!que.empty()) 35 { 36 pli now=que.top(); 37 que.pop(); 38 LL dist=now.fi; 39 int u=now.se; 40 if(vis[u]) continue; 41 vis[u]=1; 42 int sz=e[u].size(); 43 for(int i=0;i<sz;i++) 44 { 45 pli p=e[u][i]; 46 int w=p.fi; 47 int v=p.se; 48 if(!pre[v] || dis[v]>dis[u]+w) 49 { 50 dis[v]=dis[u]+w; 51 pre[v]=pre[u]; 52 que.push(mp(dis[v],v)); 53 } 54 else if(pre[u] != pre[v]) 55 ve.pb(node(pre[u],pre[v],dis[u]+dis[v]+w)); 56 } 57 } 58 return ; 59 } 60 bool cmp(node a,node b) 61 { 62 return a.w<b.w; 63 } 64 int Find(int u) 65 { 66 if(fa[u]!=u) 67 fa[u]=Find(fa[u]); 68 return fa[u]; 69 } 70 void Union() 71 { 72 sort(ve.begin(),ve.end(),cmp); 73 for(int i=1;i<=s;i++) 74 { 75 fa[fuel[i]]=fuel[i]; 76 rk[fuel[i]]=1; 77 } 78 int sz=ve.size(); 79 for(int i=0;i<sz;i++) 80 { 81 node p=ve[i]; 82 int u = p.u, v = p.v; 83 LL w = p.w; 84 u = Find(u), v = Find(v); 85 if(u==v) continue; 86 if(rk[u]<rk[v]) swap(u,v); 87 if(rk[u]==rk[v]) rk[u]++; 88 rfa[v]=u,fa[v]=u,val[v]=w; 89 } 90 return ; 91 } 92 void dealdep(int u) 93 { 94 if(dep[u]>0) return ; 95 if(fa[u]==u) 96 { 97 dep[u]=1; 98 return ; 99 } 100 dealdep(rfa[u]); 101 dep[u]=dep[rfa[u]]+1; 102 return ; 103 } 104 bool solve(int u,int v,LL b) 105 { 106 if(Find(u)!=Find(v)) return 0; 107 if(dep[u]<dep[v]) swap(u,v); 108 while(dep[u]>dep[v]) 109 { 110 if(b<val[u]) return 0; 111 u=rfa[u]; 112 } 113 if(u==v) return 1; 114 while(u!=v) 115 { 116 if(b<val[u]) return 0; 117 if(b<val[v]) return 0; 118 u=rfa[u]; 119 v=rfa[v]; 120 } 121 return 1; 122 } 123 int main() 124 { 125 scanf("%d%d%d",&n,&s,&m); 126 clr_1(dis); 127 for(int i=1;i<=s;i++) 128 { 129 scanf("%d",fuel+i); 130 que.push(mp(0,fuel[i])); 131 pre[fuel[i]]=fuel[i]; 132 dis[fuel[i]]=0; 133 } 134 for(int i=1;i<=m;i++) 135 { 136 int u,v; 137 LL w; 138 scanf("%d%d%lld",&u,&v,&w); 139 e[u].pb(mp(w,v)); 140 e[v].pb(mp(w,u)); 141 } 142 dij(); 143 Union(); 144 for(int i=1;i<=s;i++) 145 dealdep(fuel[i]); 146 int u,v; 147 LL b,maxn; 148 int q; 149 scanf("%d",&q); 150 while(q--) 151 { 152 scanf("%d%d%lld",&u,&v,&b); 153 if(solve(u,v,b)) 154 printf("TAK\n"); 155 else 156 printf("NIE\n"); 157 } 158 return 0; 159 }
View Code

4144: [AMPPZ2014]Petrol (多源最短路+最小生成樹+啟發式合並)