对于有根树T的两个结点u、v,最近公共祖先x=LCA(u,v)表示一个结点x,满足x是u、v的祖先且x的深度尽可能大。

如图,根据定义可以看出14和15的最近公共祖先是10,   15和16的最近公共祖先是1,     6和5的最近公共祖先是5......

假如我要求14和16的最近公共祖先,要怎么做呢?

最暴力的做法,就是先看14和16在不在同一层,如果他们不在同一层,那么较深的那个点往上爬(即距离根较远的那个点)

一直爬,爬到两点的深度一样

当两点深度一样时,判断他们是否在同一个点上,如果不是,则两个点同时往上爬,直到这两个点是同一个点

显然,这么做一般来说都是不行的。

下面介绍倍增法(图是用vector存的)

既然一步一步往上爬太慢了,那就一次爬远一点

我们预处理一个数组fa【x】【i】使游标快速移动,大幅度减少跳转次数。

fa【x】【i】表示 点x的第2^i个祖先

拿上图来说,fa【15】【0】就是点15的第2^0个祖先,即12(fa【15】【0】=12)

      fa【15】【1】就是点15的第2^1个祖先,即10(fa【15】【1】=10)

         fa【15】【2】就是点15的第2^2个祖先,即5(fa【15】【2】=5)

怎么预处理出这个数组呢?我们观察上图,15的第一个祖先是12,12的第一个祖先是10

即fa【15】【0】=12     fa【12】【0】=10

而15的第二个祖先是10,和12的第一个祖先是一样的

fa【15】【1】=10

也就是说,对于点15来说,15的第二个祖先就是15的第一个祖先的第一个祖先(点12的第一个祖先)

用数组表示是fa[15][1]=fa[ fa[15][0] ][ 0 ];

即fa[15][1]=fa[12][0]

推广到第2^i个祖先:递推式:fa[rt][i]=fa[fa[rt][i-1]][i-1];(rt为当前节点)

预处理代码:(depth数组存的是当前节点的深度)

void dfs(int f,int rt )//rt是当前节点,f是当前节点的父亲
{
depth[rt]=depth[f]+;
fa[rt][]=f;
for(int i=;i<;++i)//自己估计i的范围
fa[rt][i]=fa[fa[rt][i-]][i-];
for(int i=;i<E[rt].size();++i)//用vector存的图,遍历当前节点的每一个孩子
dfs(rt,E[rt][i]);
}

下一个问题是,游标移动到哪里合适?

我们的原则是,先让两个点的深度一致

然后,让两个点同时跳相同的步数

1.如果跳出根以外了,就不跳

2.如果跳到的点是相同的点,也不跳(不一定是最近的公共祖先)

不是以上两种情况,就跳

举个栗子,如图:

假如我们问14和15的lca是哪个点

按照规则

我们先让14和15在同一层上(深度一致)

14跳到了13

然后,我们看跳多少步合适,如果i=3,即2^3=8步,发现已经跳出根以外了,不跳

再看i=2, 2^2=4,13和15的第四个祖先都是5,不跳

再看i=1,即跳2步,发现都是10,不跳

再看i=0,跳一步,13跳一步到11,15跳一步到12,两个点不相同,可以跳

此时第一遍循环结束,判断这两个点的第一个祖先是不是同一个点

如果是,则找到了

如果不是,那就继续进行循环

用图表示:

第一步是14跳到13

第二步是13跳到11,15跳到12

发现11和12的第一个祖先都是10,找到答案,结束循环 (时间是log级别的)

代码:

int lca(int x,int y)//找x和y的最近公共祖先
{
if(depth[x]<depth[y])//调整x的深度大一些
swap(x,y);
while(depth[x]!=depth[y])//使x和y的深度一致
{
for(int i=;i>=;--i){//初始常数自己根据问题进行调整
if(depth[x]-(<<i)>=depth[y])
x=fa[x][i];
}
}
if(x==y) return x;
while(fa[x][]!=fa[y][])//当x和y的第一位祖先都一样时退出循环
{
for(int i=;i>=;--i){
//如果没有出界而且两点的祖先不一样,就跳
if(fa[x][i]!=&&fa[x][i]!=fa[y][i])
x=fa[x][i],y=fa[y][i];
}
}
return fa[x][];
}

代码自己写的,有点丑,网上的没看懂....

下面给出测试代码和样例,大家可以去试一试,样例就是第一幅图

#include <iostream>
#include <queue>
#include <vector>
#include <stdio.h>
using namespace std;
vector<int> E[];
int depth[];
int fa[][];
void dfs(int f,int rt )
{
depth[rt]=depth[f]+;
fa[rt][]=f;
for(int i=;i<;++i)
fa[rt][i]=fa[fa[rt][i-]][i-];
for(int i=;i<E[rt].size();++i)
dfs(rt,E[rt][i]);
}
int lca(int x,int y)
{
if(depth[x]<depth[y])//调整左边深
swap(x,y);
while(depth[x]!=depth[y])
{
for(int i=;i>=;--i){
if(depth[x]-(<<i)>=depth[y])
x=fa[x][i];
}
}
if(x==y) return x;
while(fa[x][]!=fa[y][])
{
for(int i=;i>=;--i){
if(fa[x][i]!=&&fa[x][i]!=fa[y][i])
x=fa[x][i],y=fa[y][i];
}
}
return fa[x][];
}
int main()
{
int l,r;
for(int i=;i<=;++i){
cin>>l>>r;
E[l].push_back(r);
}
dfs(,);
int a,b,t=;
while(t--)
{
cin>>a>>b;
cout<<lca(a,b)<<endl;
} return ;
}
/*
输入图:
1 2
2 4
4 5
5 6
5 7
7 10
10 11
10 12
11 13
12 15
13 14
1 3
3 8
8 9
9 16
*/

倍增法求LCA(最近公共最先)的更多相关文章

  1. 倍增法求lca(最近公共祖先)

    倍增法求lca(最近公共祖先) 基本上每篇博客都会有参考文章,一是弥补不足,二是这本身也是我学习过程中找到的觉得好的资料 思路: 大致上算法的思路是这样发展来的. 想到求两个结点的最小公共祖先,我们可 ...

  2. 倍增法求LCA

    倍增法求LCA LCA(Least Common Ancestors)的意思是最近公共祖先,即在一棵树中,找出两节点最近的公共祖先. 倍增法是通过一个数组来实现直接找到一个节点的某个祖先,这样我们就可 ...

  3. HDU 2586 倍增法求lca

    How far away ? Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)To ...

  4. 树上倍增法求LCA

    我们找的是任意两个结点的最近公共祖先, 那么我们可以考虑这么两种种情况: 1.两结点的深度相同. 2.两结点深度不同. 第一步都要转化为情况1,这种可处理的情况. 先不考虑其他, 我们思考这么一个问题 ...

  5. 在线倍增法求LCA专题

    1.cojs 186. [USACO Oct08] 牧场旅行 ★★   输入文件:pwalk.in   输出文件:pwalk.out   简单对比时间限制:1 s   内存限制:128 MB n个被自 ...

  6. 倍增法求lca:暗的连锁

    https://loj.ac/problem/10131 #include<bits/stdc++.h> using namespace std; struct node{ int to, ...

  7. 倍增法求LCA代码加详细注释

    #include <iostream> #include <vector> #include <algorithm> #define MAXN 100 //2^MA ...

  8. 浅谈倍增法求解LCA

    Luogu P3379 最近公共祖先 原题展现 题目描述 如题,给定一棵有根多叉树,请求出指定两个点直接最近的公共祖先. 输入格式 第一行包含三个正整数 \(N,M,S\),分别表示树的结点个数.询问 ...

  9. RMQ(倍增法求ST)

    解决什么问题:区间查询最值 倍增思想:每次得出结果的范围呈2的幂次增长,有人说相当于二分,目前我觉得相当于线段树的查找. 具体理解看代码: /*倍增法求ST*/ #include<math.h& ...

随机推荐

  1. arcgis 画图工具

    arcgis 4.x系列JavaScript API在4.4及以前的版本对画图工具(Draw)是不支持的,自4.5版本后才提供Draw的类.相关连接:https://developers.arcgis ...

  2. WPF中使用DataGrid时操作列按钮问题

    在使用DataGrid的过程中,我们有时候需要对选取的某一行数据进行多个操作,这个时候操作列只有一个按钮显然无法满足我们的要求,我们需要多个按钮才能达到我们的目的. UI页面代码: <Grid& ...

  3. spring boot中使用@Async实现异步调用任务

    本篇文章主要介绍了spring boot中使用@Async实现异步调用任务,小编觉得挺不错的,现在分享给大家,也给大家做个参考.一起跟随小编过来看看吧 什么是“异步调用”? “异步调用”对应的是“同步 ...

  4. php魔术变量以及命名空间

    魔术变量: PHP 向它运行的任何脚本提供了大量的预定义常量. 不过很多常量都是由不同的扩展库定义的,只有在加载了这些扩展库时才会出现,或者动态加载后,或者在编译时已经包括进去了. 有八个魔术常量它们 ...

  5. [Android] 对于com.google.gson.JsonElement的转义问题

    不多说了,com.google.gson.JsonElement使用的时候,toString()跟getAsString()这两个方法对于特殊字符的转义是不同的, 看这里的解释: https://st ...

  6. [NOIP2017赛前复习第二期]复赛考试技巧与模版-普及组

    考试技巧 1.拿到考卷首先通看题目,按自己感觉的难度排序(普及一般是1-2-3-4了~还是相信出题人不会坑我们的2333) 2.一般来说,普及组前两道题比较简单(大水题啊233~),但是通常坑很多,例 ...

  7. 【kafka学习之六】kakfa消息生产、消费示例

    环境 虚拟机:VMware 10 Linux版本:CentOS-6.5-x86_64 客户端:Xshell4 FTP:Xftp4 jdk1.8 kafka_2.11-0.11.0.0 zookeepe ...

  8. 解决SVN提交和更新代码冲突?

    解决冲突有三种选择: 1.放弃自己的更新,使用svn revert(回滚),然后提交.在这种方式下不需要使用svn resolved(解决) 2.放弃自己的更新,使用别人的更新.使用最新获取的版本覆盖 ...

  9. javascript声明变量

    <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title> ...

  10. flutter -icons

    icon 合集 https://material.io/tools/icons/?icon=account_balance&style=baseline