0%

P3379 【模板】最近公共祖先(LCA)(树链剖分做法)

题目大意

给定一棵有根多叉树,包含 NN 个节点,树根为节点 SS。有 MM 次询问,每次询问给定两个节点 aabb,求它们的最近公共祖先(LCA,Lowest Common Ancestor)。

输入格式:

  • 第一行:NN MM SS(节点数、询问数、根节点)

  • 接下来 N1N-1 行:每行两个整数 xx yy,表示 xxyy 之间有一条边

  • 接下来 MM 行:每行两个整数 aa bb,表示询问 aabb 的最近公共祖先

输出格式:

  • MM 行,每行一个整数,表示对应询问的答案

思路讲解

重点在于这两个dfs,其实还挺简单的。dfs负责找出重儿子,dfss负责重链剖分。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
void dfs(int u){	// 找出一个节点的重儿子
siz[u]=1;
dep[u]=dep[fa[u]]+1;
for(auto &v:g[u]){
if(v==fa[u]) continue;
fa[v]=u;
dfs(v);
siz[u]+=siz[v];
if(siz[v]>siz[son[u]]){
son[u]=v;
}
}
}
void dfss(int u,int up){ // 维护top节点
top[u]=up;
if(son[u]){
dfss(son[u],up);
}
for(auto &v:g[u]){
if(v==son[u]) continue;
if(v==fa[u]) continue;
// 是一条新的重链
dfss(v,v);
}
}

lca的求法。

1
2
3
4
5
6
7
8
9
10
11
int lca(int u,int v){
while(top[u]!=top[v]){ // 判断两点在不在一条链上面
if(dep[top[u]]>dep[top[v]]){ // 要跳到更深的链上面
// 必须要加一个fa,否则就走不出这条链了,死循环了
u=fa[top[u]];
}else{
v=fa[top[v]];
}
}
return dep[u]<dep[v]?u:v;
}

AC代码

心路历程(WA,TLE,MLE……)