写在前面

目录

一、LCA的定义

二、暴力法求LCA

三、倍增法求LCA

四、树链剖分求LCA

五、LCA典型例题

题目完成度


一、LCA的定义

LCA指的是最近公共祖先。具体地,给定一棵有根树,若结点z既是结点x的祖先,又是结点y的祖先,则称z是x,y的公共祖先。在x,y的公共祖先中,深度最大的一个结点称为x,y的最近公共祖先,记为LCA(x,y)

go back


、暴力法求LCA

暴力法,顾名思义,非常暴力,这里简单介绍一下

先DFS一遍找出每个点的深度,然后先从深度大的往上跳,跳到x,y两个点深度相同。

如果发现此时x和y是同一个结点,那么原本深度小的那个结点就是两个点的最近公共祖先。

如果不是同一个点,那么就两个点一起同时往上跳,直到发现两个点相同,则此时到达的这个结点为x,y两点的最近公共祖先。

484很简单?QWQ

那我等下放个代码,就酱紫吧

int LCA(int x,int y){
if(dep[x]<dep[y]) swap(x,y);//默认x深度大于y
while(dep[x]>dep[y]) x=fa[x];//x每次变成自己的父结点,即往上跳一步
if(x==y) return y;
//如果此时两个结点相同,那么原本深度较小的结点就是LCA
while(x!=y)
x=fa[x],y=fa[y];//两个点同时一步步往上跳
return x;//此时两个节点相同,都是LCA
}

其实还有一种暴力的方法

就是先让其中一个点一路跳到根结点,标记经过的结点,然后另一个点也往上跳,遇到的第一个标记了的结点就是LCA

同样放下代码

int LCA(int x,int y){
while(x!=root){//root为根结点
x=fa[x];
visit[x]=;//标记
}
while(!visit[y])//如果遇到被标记了的就找到了LCA
y=fa[y];
return y;
}

go back


三、倍增法求LCA

这是一个非常重要的算法啦!一定要记住QAQ

我来讲一讲啦

【预处理】

设f[x][k]表示x结点的2k辈祖先,即从x向根结点走2k步到达的结点

如果该结点不存在,则令f[x][k]=0,f[x][0]就是x的父亲结点

因为2k=2k-1+2k-1,所以对于1≤k≤logn,有f[x][k]=f[f[x][k-1]][k-1]

预处理部分的时间复杂度为O(nlongn)

void ready(int x,int father){
dep[x]=dep[father]+;//计算深度
go(i,,)//预处理f数组
f[x][i+]=f[f[x][i]][i];
for(int i=head[x];i;i=next[i]){
int y=to[i];//用链式前向星存边
if(y==father) continue;
f[y][]=x;
ready(y,x);
}
}

【查找LCA】

预处理过后可以对多个x,y查找LCA,每次的时间复杂度均为O(nlogn)

具体操作如下:

1.假设dep[x]≥dep[y],如果不成立可以交换

2.用二进制拆分思想把x上调到与y同一深度

其实就是依次尝试让x向上跳k=2logn...21,20,若到达的结点比y深,则令x=f[x][k]

3.和上面说的一样,若此时x与y相等,则y就是LCA

4.若此时x≠y,那么x和y同时往上跳,同样用二进制拆分思想,把x和y同时向上跳,并保持深度一致且不相会

与上面调整x时一样,让x和y尝试向上走k=2logn...21,20步,若f[x][k]≠f[y][k](即两点不相会),则令x=f[x][k],y=f[y][k]

5.结束时x和y必然只差一步就相会了,所以他们的父节点f[x][0]就是LCA啦!QWQ

int LCA(int x,int y){
if(dep[x]<dep[y]) swap(x,y);
back(i,,){
if(dep[f[x][i]]>=dep[y]) x=f[x][i];
//只要x的深度没有比y小就可以继续跳
if(x==y) return x;
}
back(i,,){
if(f[x][i]!=f[y][i])//跳的过程中要保证两点不相会
x=f[x][i],y=f[y][i];
}
return f[x][];
}

最后再放一个完整版代码吧QWQ

 #include<bits/stdc++.h>
#define go(i,a,b) for(register int i=a;i<=b;i++)
#define back(i,a,b) for(register int i=a;i>=b;i--)
using namespace std;
const int N=;
int n,m;
int dep[N],head[N];
int f[N][];
int next[N*],to[N*],num=;
int last=;
int fr(){
int w=,q=;
char ch=getchar();
while(ch<''||ch>''){
if(ch=='-') q=-;
ch=getchar();
}
while(ch<=''&&ch>='')
w=(w<<)+(w<<)+ch-'',ch=getchar();
return w*q;
}
void ready(int x,int father){
dep[x]=dep[father]+;
go(i,,)
f[x][i+]=f[f[x][i]][i];
for(int i=head[x];i;i=next[i]){
int y=to[i];
if(y==father) continue;
f[y][]=x;
ready(y,x);
}
}
int LCA(int x,int y){
if(dep[x]<dep[y]) swap(x,y);
back(i,,){
if(dep[f[x][i]]>=dep[y]) x=f[x][i];
if(x==y) return x;
}
back(i,,){
if(f[x][i]!=f[y][i])
x=f[x][i],y=f[y][i];
}
return f[x][];
}
int main(){
n=fr();
int root,father;
go(i,,n){
father=fr();
if(father==) root=i;
next[++num]=head[father];
to[num]=i;
head[father]=num;
next[++num]=head[i];
to[num]=father;
head[i]=num;
}
ready(root,);
m=fr();
while(m--){
int x=fr(),y=fr();
last=LCA(x,y);
printf("%d\n",last);
}
return ;
}

代码戳这里

go back


四、树链剖分求LCA

咕咕咕咕咕

go back


五、LCA典型例题

咕咕咕咕

go back

LCA学习笔记的更多相关文章

  1. 倍增LCA学习笔记

    前言 ​ "倍增",作为一种二进制拆分思想,广泛用于各中算法,如\(ST\)表,求解\(LCA\)等等...今天,我们仅讨论用该思想来求解树上两个节点的\(LCA\)(最近公共祖先 ...

  2. 倍增求LCA学习笔记(洛谷 P3379 【模板】最近公共祖先(LCA))

    倍增求\(LCA\) 倍增基础 从字面意思理解,倍增就是"成倍增长". 一般地,此处的增长并非线性地翻倍,而是在预处理时处理长度为\(2^n(n\in \mathbb{N}^+)\ ...

  3. 树链剖分 树剖求lca 学习笔记

    树链剖分 顾名思义,就是把一课时分成若干条链,使得它可以用数据结构(例如线段树)来维护 一些定义: 重儿子:子树最大的儿子 轻儿子:除了重儿子以外的儿子 重边:父节点与重儿子组成的边 轻边:除重边以外 ...

  4. kruskal重构树学习笔记

    \(kruskal\) 重构树学习笔记 前言 \(8102IONCC\) 中考到了,本蒟蒻不会,所以学一下. 前置知识 \(kruskal​\) 求最小(大)生成树,树上求 \(lca​\). 算法详 ...

  5. 仙人掌&圆方树学习笔记

    仙人掌&圆方树学习笔记 1.仙人掌 圆方树用来干啥? --处理仙人掌的问题. 仙人掌是啥? (图片来自于\(BZOJ1023\)) --也就是任意一条边只会出现在一个环里面. 当然,如果你的图 ...

  6. Day 4 学习笔记 各种图论

    Day 4 学习笔记 各种图论 图是什么???? 不是我上传的图床上的那些垃圾解释... 一.图: 1.定义 由顶点和边组成的集合叫做图. 2.分类: 边如果是有向边,就是有向图:否则,就是无向图. ...

  7. 树上启发式合并(dsu on tree)学习笔记

    有丶难,学到自闭 参考的文章: zcysky:[学习笔记]dsu on tree Arpa:[Tutorial] Sack (dsu on tree) 先康一康模板题吧:CF 600E($Lomsat ...

  8. 「学习笔记」wqs二分/dp凸优化

    [学习笔记]wqs二分/DP凸优化 从一个经典问题谈起: 有一个长度为 \(n\) 的序列 \(a\),要求找出恰好 \(k\) 个不相交的连续子序列,使得这 \(k\) 个序列的和最大 \(1 \l ...

  9. OI知识点|NOIP考点|省选考点|教程与学习笔记合集

    点亮技能树行动-- 本篇blog按照分类将网上写的OI知识点归纳了一下,然后会附上蒟蒻我的学习笔记或者是我认为写的不错的专题博客qwqwqwq(好吧,其实已经咕咕咕了...) 基础算法 贪心 枚举 分 ...

随机推荐

  1. 在app中从下向上滑动,以找到不在默认第一页的元素

    1.在app中,如果第一面没有出现你要的元素,需要下滑拉动到你需要的元素的位置

  2. iOS开发基础-九宫格坐标(4)

    对iOS开发基础-九宫格坐标(3)的代码进行进一步优化. 新建一个 UIView 的子类,并命名为 WJQAppView ,将 appxib.xib 中的 UIView 对象与新建的视图类进行关联. ...

  3. Java 开发笔记2

    Java获取参数名称 https://blog.csdn.net/z69183787/article/details/81117525 DefaultParameterNameDiscoverer() ...

  4. lvs--小白博客

    lvs 一.负载均衡LVS基本介绍 LVS是 Linux Virtual Server 的简称,也就是Linux虚拟服务器.这是一个由章文嵩博士发起的一个开源项目,它的官方网站是 http://www ...

  5. 如何查杀stopped进程

    在Linux系统下面,top命令可以查看查看stopped进程.但是不能查看stopped进程的详细信息.那么如何查看stopped 进程,并且杀掉这些stopped进程呢? ps -e j | gr ...

  6. java中内存分配

    java程序运行时内存分配详解  一. 基本概念 每运行一个java程序会产生一个java进程,每个java进程可能包含一个或者多个线程,每一个Java进程对应唯一一个JVM实例,每一个JVM实例唯一 ...

  7. 其它综合-运维老鸟分享linux运维发展路线规划

    运维老鸟分享linux运维发展路线规划 linux 运维发展路线常见的就是下面两条路线: 第一条:运维应用-->系统架构-->运维开发-->系统开发 第二条:运维应用-->应用 ...

  8. AngularJS 1.x系列:Node.js安装及npm常用命令(1)

    1. Node.js安装 1.1 Node.js下载 Node.js官网:https://nodejs.org 当前下载版本(含npm):Latest LTS Version: v6.10.3 (in ...

  9. Python——Set集合

    一.定义 Set集合用于表示相互之间无需的一种组合对象,包括:并集.交集.补集 二.集合的两种模式 sample = set()  初始化普通集合 sample = frozenset() 初始化不可 ...

  10. python 条件分支与循环

    一.if判断: 语法一: if 条件: # 条件成立时执行的子代码块 代码1 代码2 代码3 示例: sex='female' age=18 is_beautiful=True if sex == ' ...