D11 最近公共祖先(LCA)树链剖分
D11 最近公共祖先(LCA)树链剖分_哔哩哔哩_bilibili
1. dfs1(u,fa) 处理 fa[],son[],dep[],siz[]
2. dfs2(u,top) 处理 top[]。搜重儿子;搜轻儿子
3. lca(u,v) 链头不同则上爬,同链取dep小
// 树链剖分 O(mlogn) #include<bits/stdc++.h> using namespace std; const int N=500010; int n,m,s,a,b; vector<int> e[N]; int fa[N],son[N],dep[N],siz[N],top[N]; void dfs1(int u,int f){ //搞fa,son,dep,siz fa[u]=f;dep[u]=dep[f]+1;siz[u]=1; for(int v:e[u]){ if(v==f) continue; dfs1(v,u); siz[u]+=siz[v]; if(siz[son[u]]<siz[v]) son[u]=v; } } void dfs2(int u,int t){ //搞top top[u]=t; //记录链头 if(son[u]) dfs2(son[u],t); //搜重儿子 for(int v:e[u]){ if(v==fa[u]||v==son[u])continue; dfs2(v,v); //搜轻儿子 } } int lca(int u,int v){ while(top[u]!=top[v]){ if(dep[top[u]]<dep[top[v]])swap(u,v); u=fa[top[u]]; } return dep[u]<dep[v]?u:v; } int main(){ scanf("%d%d%d",&n,&m,&s); for(int i=1; i<n; i++){ scanf("%d%d",&a,&b); e[a].push_back(b); e[b].push_back(a); } dfs1(s,0); dfs2(s,s); while(m--){ scanf("%d%d",&a,&b); printf("%d\n",lca(a,b)); } }
浙公网安备 33010602011771号