【模板时间】◆模板·I◆ 倍增计算LCA

【模板·I】LCA(倍增版)

既然是一篇重点在于介绍、分析一个模板的Blog,作者将主要分析其原理,可能会比较无趣……(提供C++模板)

另外,给reader们介绍另外一篇非常不错的Blog(我就是从那篇博客开始自学LCA的):+LCA-by 殇雪+


一、原理

 LCA即最近公共祖先,一般用LCA(u,v)表示u、v的最近公共祖先。举个例子:

LCA举例

(Tab:以下“树”均指有根树)由于在树中,除根节点的每个节点都有且仅有一个父节点,我们很容易得到一个结论——u,v的最近公共祖先的任意祖先一定也是u,v的公共祖先。假设x是u,v的公共祖先,LCA(u,v)≤x。倍增求LCA的基础是 2进制能够表示任意整数 。所以令u,v的最近公共祖先与u、v分别距离lu、lv,并可以将其表示成2进制。

设 dfu[v][k] 表示 节点v向上寻找到的第2k代祖先(比如之前的图中,dfu[4][1]=1),dep[v] 表示 v的深度(根节点的深度依照题目定为1或0)。

不妨设 dep[v]>dep[u] 。首先要使u、v同层,即dep[u]=dep[v],可以通过将v上移到它的 dep[u]-dep[v] 代祖先来实现,再将dep[v]-dep[u]转为2进制,就可以通过dfu实现。

u、v同层后,设它们距离最近公共祖先l个单位。同样,l也可以表示为2进制,也就可以通过dfu解决。

由于每一次移动都是在2进制下进行,求LCA的复杂度大约可看为 O(log2n)

二、算法实现

①大致步骤:

初始化:  DFS初始化每个点v的dep[v]以及直接父亲(dfu[v][0]);

         递推计算全部dfu;

计算LCA: 上移u、v至同一层;

 同时上移u、v找最近公共祖先;

②初始化:

DFS可以通过参数下传父亲节点以及节点深度(eg:void DFS(int u,int fa,int depth))。(建议用邻接表的方式)遍历每一个儿子,同时初始化。

根据dfu定义,dfu[v][i+1]表示v的第2i+1代祖先,也就是第(2i+2i)代祖先。则可以先找到v的第2i代祖先u,再找到u的第2i代祖先。递推式如下:

dfu[v][i+1]=dfu[dfu[v][i]][i]; //dfu[v][i]表示v的第2i代祖先

由于递推过程中,dfu[v][i]已经计算出来了,我们就可以从dfu[v][0]出发推出所有dfu。

③计算LCA(u,v)

为了方便计算,先保证dep[v]>dep[u]。

要将u、v移动到同层,即把v上移(dep[u]-dep[v])层。由于我们知道v的2k代祖先,我们可以把dep[u]-dep[v]拆分成 2a1+2a2+...+2ap(C++实现可以判断 (dep[v]-dep[u])>>i&1,即2进制的(dep[v]-dep[u])的第i位是否为1),按次上移即可。

如果此时u=v,则说明最初u是v的祖先,则LCA(u,v)=u。

除开上述u=v的情况。由于 LCA(u,v) 的祖先一定也是u,v的公共祖先,所以我们可以将u、v上移到LCA(u,v)的下一层,即u,v的父亲为LCA(u,v)。从高到低枚举i(i最大为ceil(log2(n)),即退化为链后根与叶子节点),如果dfu[u][i]==dfu[v][i],则说明dfu[u][i]已经是LCA(u,v)或层数已经高于LCA(u,v)了,由于无法直接判断是否是LCA(u,v),我们就可以选择不上移(目标是将u,v转移到LCA(u,v)的下一层);如果dfu[u][i]!=dfu[v][i],则说明还没有到LCA(u,v),就可以上移。最后就可以移动到LCA(u,v)的下一层。

其实上述操作无非是令u,v到LCA(u,v)的距离为S,将S-1表示为2进制,再通过dfu顺次上移。最后得到dfu[u][0](或者dfu[v][0])就是LCA(u,v)了。

三、C++代码

初始化:

 1 void DFS(int u,int fa,int depth)
 2 {
 3     dfu[u][0]=fa;dep[u]=depth; //更新v的父节点以及深度
 4     for(int i=0;i<lnk[u].size();i++) //lnk是vector的邻接表
 5         DFS(lnk[u][i],u,depth+1);
 6 }
 7 void Prepare()
 8 {
 9     DFS(0,-1,1); //先处理出节点的深度(dep)和直接祖先(dfu[0])
10     for(int i=0;i+1<20;i++)
11         for(int j=0;j<n;j++)
12             if(dfu[j][i]<0) dfu[j][i+1]=-1; //上移位置已经超过根节点
13             else dfu[j][i+1]=dfu[dfu[j][i]][i];
14 }
View Code

计算LCA:

 1 int LCA(int u,int v)
 2 {
 3     if(dep[u]>dep[v]) swap(u,v); //保证u不高于v
 4     for(int i=0;i<20;i++) //拆分二进制
 5         if(((dep[v]-dep[u])>>i)&1) //上移到同一层
 6             v=dfu[v][i];
 7     if(u==v) return u; //u最初是v的根节点
 8     for(int i=19;i>=0;i--)
 9         if(dfu[u][i]!=dfu[v][i])
10             u=dfu[u][i],v=dfu[v][i];
11     return dfu[u][0];
12 }
View Code

The End

Thanks for reading!

- Lucky_Glass

(Tab:如果我有没讲清楚的地方可以直接在邮箱[email protected] email我,在周末我会尽量解答并完善博客~)

猜你喜欢

转载自www.cnblogs.com/LuckyGlass-blog/p/9325629.html