1 solutions
-
1thakuremon LV 4 @ 2024-12-10 19:30:59
Root the tree at any node lets say at 1.
For each query we just need to find two special node \(X,Y\) to answer the query.\(Dis\) : shortest distance of node \(C\) and \(P\)
\(X\) : the furthest node from \(C\) in the path of \(C\) to \(P\) where terrorists can reach safely before police. This node must be the node at distance \([(Dis-1)/2]\) from node \(C\).\(Y\) : the nearest node from \(C\) in the path of \(C\) to \(P\) where terrorists will be caught if they try to visit this node. This node must be the node at distance \([(Dis-1)/2]+1\). Basically this is the next node of \(X\) in the path of \(C\) to \(P\)
\(X\) and \(Y\) can be found easily using lowest common ancestor and \(k-th\) parent in \(log(N)\) time.
Now if we remove the edge between node \(X\) and \(Y\) we will have 2 separate tree lets name them \(X-tree\) and \(Y-tree\).
Terrorists can visit any node in \(X-tree\) before police but if they try to visit any node in \(Y-tree\) then they must go through \(Y\) node first and by definition of the \(Y\) node they will be get arrested in \(Y\) node.So finally we just need to check if \(X-tree\) contains at least one vulnerable node or not so that the terrorists can escape.
We can find the number of vulnerable nodes in \(X-tree\) and \(Y-tree\) using subtree sum in constant time.total complexity: \(Q*log(N)\)
code: thakuremon
/* * Copyright (c) 2024 Emon Thakur * All rights reserved. */ #include<bits/stdc++.h> using namespace std; int a[200005],vulcnt[200005],depth[200005]; vector<int> g[200005]; void precal(int node,int par,int dep) { if(a[node]) vulcnt[node]=1; depth[node] = dep; for(auto e:g[node]) { if(e==par) continue; precal(e,node,dep+1); } for(auto e:g[node]) { if(e==par) continue; vulcnt[node] += vulcnt[e]; } } struct LCA{ int dp[200005][20]; void binarylifting(int node,int par) { dp[node][0] = par; for(int i=1;i<20;i++) dp[node][i] = dp[dp[node][i-1]][i-1]; for(auto e:g[node]) { if(e == par) continue; binarylifting(e,node); } } int kthparent(int node,int k) { for(int i=19;i>=0;i--) { if(k & (1<<i)) node = dp[node][i]; } return node; } int lca(int u,int v) { if(depth[u] < depth[v]) swap(u,v); u = kthparent(u , depth[u]-depth[v]); if(u == v) return v; for(int i=19;i>=0;i--) { if(dp[u][i] == dp[v][i]) continue; u = dp[u][i]; v = dp[v][i]; } return dp[u][0]; } int dist(int u,int v) { int lcaa = lca(u,v); return depth[u]-depth[lcaa] + depth[v]-depth[lcaa]; } int nodeatdis(int u,int v,int d) { if(d <= depth[u]-depth[lca(u,v)]) return kthparent(u,d); return kthparent(v,dist(u,v)-d); } }lca; int main() { int n; cin >> n; for(int i=1;i<=n;i++) cin >> a[i]; for(int i=0;i<n-1;i++) { int u,v; cin >> u >> v; g[u].push_back(v); g[v].push_back(u); } precal(1,0,0); lca.binarylifting(1,0); int q; cin >> q; while(q--) { int c,p; cin >> c >> p; if(c==p) {cout<<"weee"<<endl; continue;} int dis = lca.dist(c,p); int x = lca.nodeatdis(c,p,(dis-1)/2); int y = lca.nodeatdis(c,p,((dis-1)/2)+1); if(depth[x]>depth[y]) { if(vulcnt[x]) cout<<"oops"<<endl; else cout<<"weee"<<endl; } else { if(vulcnt[1]-vulcnt[y]) cout<<"oops"<<endl; else cout<<"weee"<<endl; } } }
- 1
Information
- ID
- 1134
- Difficulty
- 8
- Category
- (None)
- Tags
- (None)
- # Submissions
- 59
- Accepted
- 8
- Accepted Ratio
- 14%
- Uploaded By