最近公共祖先 LCA 倍增写法

LCA的倍增主要由三个重要的过程组成

预处理lg数组

DFS求fa depth

倍增节点

观看以下内容前建议先把完整代码大致纵览一遍,有利于理解各个函数的意义

倍增思想

暴力解决LCA是通过 x 和 y 一个一个的往上跳

而倍增的思想是希望节点能够一次性尽可能的多跳

并且可以通过一个有序数列来控制跳的层数

1 2 4 8 16 32 ...

\(2^0\) \(2^1\) \(2^2\) \(2^3\) \(2^4\) \(2^5\) ...

(可以有机联想10进制到2进制的转化)

比如: 当前需要跳35层

35 - 32 = 3

3 - 2 = 1

1 - 1 = 0

这样就将原本需要跳35次压缩到只需要跳3次

那如何知道应该跳几次呢?

这时候就要通过lg数组来实现

预处理lg数组

lg数组的含义: \(lg[i]\) 代表深度为i的节点一次性可以往上最多跳 2(lg[i]-1) 个节点

在 Read() 中,我们可以找到如下代码:

	for(int i = 1;i <= n;i++){
lg[i] = lg[i-1] + (1 << lg[i-1] == i);
}

为了理解这段话的意思 我们先把这个处理后的数组 lg[0 - 100] 打印出来:

0 1 2 2 3 3 3 3 4 4 4 4 4 4 4 4 5 5 5 5 5 5 5 5 5 5 5 5 5 5 5 5 6 6 6 6 6 6 6 6 6 6 6 6 6 6 6 6 6 6 6 6 6 6 6 6 6 6 6 6 6 6 6 6 7 7 7 7 7 7 7 7 7 7 7 7 7 7 7 7 7 7 7 7 7 7 7 7 7 7 7 7 7 7 7 7 7 7 7 7 7

比如: 当预期要跳的深度为35 那就看 lg[35] = 6

则需要跳的次数为 26-1 = 32

所以最多跳32层

其余也是同理

DFS求fa depth

Code

void dfs(int u,int f){
fa[u][0] = f;
depth[u] = depth[f] + 1;
for(int i = 1;i <= lg[depth[u]];i++)
fa[u][i] = fa[fa[u][i-1]][i-1];
for(int i = head[u];i;i = e[i].u){
if(e[i].v != f) dfs(e[i].v,u);
}
}

求depth

我们都知道DFS是从上往下搜的

可以通过父节点的depth+1来得到当前节点的depth

因此处理depth只需要将父节点的值加1即可:

	depth[u] = depth[f] + 1;

求fa

要求fa数组要先搞懂其含义:\(fa[i][j]\) 指 节点 \(i\) 的 \(2^j\)级祖先

由于 \(2^j\) = 2j-1 + 2j-1

所以 \(i\) 的 2j-1 级祖先的 \(j-1\) 级祖先就是 \(i\) 的 \(j\) 级祖先:

	fa[u][i] = fa[fa[u][i-1]][i-1];

既然要通过前面的fa求后面的fa

那必须将\(fa[i][0]\)求出 既 \(i\) 的父节点:

	fa[u][0] = f;

倍增节点

Code

int LCA(int x,int y){
if(depth[x] < depth[y]) swap(x,y);//让 节点x 作为较深的节点
while(depth[x] > depth[y]){//把 x 和 y 放至相同深度
x = fa[x][lg[depth[x]-depth[y]] - 1];
}
if(x == y) return y;/*
如果 x 被拉到 y 上
说明 y 就是 x 的祖先
因此直接返回 y
*/
for(int k = lg[depth[x]] - 1;k >= 0;k--){
if(fa[x][k] != fa[y][k])//如果 x 和 y 没有相遇 则说明可以倍增到这个位置
x = fa[x][k],y = fa[y][k];
}
return fa[x][0];//返回 x 的父亲节点
}

注释中写道 如果 x 和 y 没有相遇 则说明可以倍增到这个位置

这是什么意思?

我们不妨设想如果没有这个if()

直接相遇就break;

那就会发生非最近公共祖先的问题(就是倍增过头了)

因此一直倍增却不允许相遇 就会让他们停在LCA的下一层

这一部分比较简单 所以就直接把注释打在代码上了

完整代码

//P3379 注释版
#include<bits/stdc++.h>
#define maxn 500010
using namespace std; int head[maxn],cnt;
struct tree{
int u,v;
tree(int a = 0,int b = 0){
u = head[a];
v = b;
}
}e[maxn << 1];
void add(int u,int v){
e[++cnt]=tree(u,v);
head[u] = cnt;
e[++cnt]=tree(v,u);
head[v] = cnt;
}//邻接表 int depth[maxn],fa[maxn][22],lg[maxn];
int m,n,s;
/*
定义变量
depth[i] 节点i的深度
fa[i][j] 节点i的2^J级祖先
lg[i] 辅助参数
*/ void Read(){//输入
cin >> n >> m >> s;
int x,y;
for(int i = 1;i < n;i++){
cin >> x >> y;
add(x,y);
}
for(int i = 1;i <= n;i++){
lg[i] = lg[i-1] + (1 << lg[i-1] == i);
}/*
常数优化
lg[i]代表深度为i的节点可以往上最多跳2^(lg[i] - 1)个节点
*/
} void dfs(int u,int f){
fa[u][0] = f;
depth[u] = depth[f] + 1;
for(int i = 1;i <= lg[depth[u]];i++)
fa[u][i] = fa[fa[u][i-1]][i-1];
for(int i = head[u];i;i = e[i].u){
if(e[i].v != f) dfs(e[i].v,u);
}
}/*
DFS
可以求出 fa depth
*/ int LCA(int x,int y){
if(depth[x] < depth[y]) swap(x,y);//让 节点x 作为较深的节点
while(depth[x] > depth[y]){//把 x 和 y 放至相同深度
x = fa[x][lg[depth[x]-depth[y]] - 1];
}
if(x == y) return y;/*
如果 x 被拉到 y 上
说明 y 就是 x 的祖先
因此直接返回 y
*/
for(int k = lg[depth[x]] - 1;k >= 0;k--){
if(fa[x][k] != fa[y][k])//如果 x 和 y 没有相遇 则说明可以倍增到这个位置
x = fa[x][k],y = fa[y][k];
}
return fa[x][0];//返回 x 的父亲节点
} int main(){
Read();
dfs(s,0);
for(int i = 1;i <= m;i++){
int x,y;
cin >> x >> y;
cout << LCA(x,y) <<endl;
}
return 0;
}

[C++]P3379 LCA 最近公共祖先的更多相关文章

  1. lca 最近公共祖先

    http://poj.org/problem?id=1330 #include<cstdio> #include<cstring> #include<algorithm& ...

  2. Tarjan算法应用 (割点/桥/缩点/强连通分量/双连通分量/LCA(最近公共祖先)问题)(转载)

    Tarjan算法应用 (割点/桥/缩点/强连通分量/双连通分量/LCA(最近公共祖先)问题)(转载) 转载自:http://hi.baidu.com/lydrainbowcat/blog/item/2 ...

  3. LCA(最近公共祖先)模板

    Tarjan版本 /* gyt Live up to every day */ #pragma comment(linker,"/STACK:1024000000,1024000000&qu ...

  4. CodeVs.1036 商务旅行 ( LCA 最近公共祖先 )

    CodeVs.1036 商务旅行 ( LCA 最近公共祖先 ) 题意分析 某首都城市的商人要经常到各城镇去做生意,他们按自己的路线去做,目的是为了更好的节约时间. 假设有N个城镇,首都编号为1,商人从 ...

  5. LCA近期公共祖先

    LCA近期公共祖先 该分析转之:http://kmplayer.iteye.com/blog/604518 1,并查集+dfs 对整个树进行深度优先遍历.并在遍历的过程中不断地把一些眼下可能查询到的而 ...

  6. LCA 近期公共祖先 小结

    LCA 近期公共祖先 小结 以poj 1330为例.对LCA的3种经常使用的算法进行介绍,分别为 1. 离线tarjan 2. 基于倍增法的LCA 3. 基于RMQ的LCA 1. 离线tarjan / ...

  7. LCA 最近公共祖先 (笔记、模板)

    求lca的方法大体有三种: 1.dfs+RMQ(线段树 ST表什么的) 在线 2.倍增 在线 3.tarjan 离线 ps:离线:所有查询全输入后一次解决 在线:有一个查询输出一次 以下模板题为 洛谷 ...

  8. LCA最近公共祖先---倍增法笔记

    先暂时把模板写出来,A几道题再来补充 此模板也是洛谷上的一道模板题 P3379 [模板]最近公共祖先(LCA) #pragma GCC optimize(2) //o2优化 #include < ...

  9. 【图论算法】LCA最近公共祖先问题

    LCA模板题https://www.luogu.com.cn/problem/P3379题意理解 对于有根树T的两个结点u.v,最近公共祖先LCA(u,v)表示一个结点x,满足x是u.v的祖先且x的深 ...

  10. LCA最近公共祖先 ST+RMQ在线算法

    对于一类题目,是一棵树或者森林,有多次查询,求2点间的距离,可以用LCA来解决.     这一类的问题有2中解决方法.第一种就是tarjan的离线算法,还有一中是基于ST算法的在线算法.复杂度都是O( ...

随机推荐

  1. Python 运行 shell 命令的一些方法

    哈喽大家好,我是咸鱼 我们知道,python 在自动化领域中被广泛应用,可以很好地自动化处理一些任务 就比如编写 Python 脚本自动化执行重复性的任务,如文件处理.数据处理.系统管理等需要运行其他 ...

  2. C语言基础-基础指针

    文章目录 指针 前言 1.什么是指针 2.指针的使用 (1)指针的定义 (2)指针的赋值 (3)指针类型 (4)如何使用指针 3.野指针 (1)导致野指针的原因 ① 未初始化指针 ②指针越界访问 ③指 ...

  3. 我学到的一下vue使用技巧

    这两天学到的vue使用技巧 v-if , 当封装组件的时候,用到的props,最外层最好加个v-if,防止出现cannot read property of undefined 这样的错误,如果pro ...

  4. ChatGPT 助力开发人员改进代码的5个方式

    近年来,在软件开发中使用人工智能和机器学习变得越来越普遍.因此,开发人员开始转向像 OpenAI 的 ChatGPT 这样的工具来简化他们的工作,提高他们的工作效率.ChatGPT是一个由 OpenA ...

  5. Redis系列18:过期数据的删除策略

    Redis系列1:深刻理解高性能Redis的本质 Redis系列2:数据持久化提高可用性 Redis系列3:高可用之主从架构 Redis系列4:高可用之Sentinel(哨兵模式) Redis系列5: ...

  6. [碎碎念]和ljf老师聊天得到的一些启发,希望大家一起来吹水

    关于写这个小文 和ljf老师聊天得到的一些启发,希望能够总结出来方便回顾,并且我觉得这些想法有一定的普适性,可以供大家参考. 疑问 我的疑问是,我现在主要在做fuzz+pwn,能够进行漏洞挖掘,以及w ...

  7. SpringBoot项目统一处理返回值和异常

    目录 简介 前期准备 统一封装报文 统一异常处理 自定义异常信息 简介 当使用SpringBoot开发Web项目的API时,为了与前端更好地通信,通常会约定好接口的响应格式.例如,以下是一个JSON格 ...

  8. 《SQL与数据库基础》09. 事务

    @ 目录 事务 简介 操作 方式一 方式二 四大特性(ACID) 并发事务问题 事务隔离级别 本文以 MySQL 为例 事务 简介 事务是一组操作的集合,它是一个不可分割的工作单位.事务会把所有的操作 ...

  9. 白话领域驱动设计DDD

    容我找个借口先,日常工作太忙,写作略有荒废.一直想聊下领域驱动设计,以下简称DDD,之前也看过一些教程,公司今年两个项目--银行核心和信用卡核心,都深度运用DDD成功落地,有人说DDD挺难理解,在此讲 ...

  10. shopee商品详情接口的应用

    Shopee是东南亚和台湾地区最大的电子商务平台之一,成立于2015年,目前覆盖6个国家和地区.作为一家新兴电商平台,Shopee拥有快速增长的销售额和庞大的用户群体,为开发者提供了丰富的商业机会.其 ...