lca最近公共祖先,是指两个点最近的祖先节点;求lca我知道的有三种倍增,
st表,tarjan,我要介绍的是倍增,我才不会告诉你我只会这一个。
话说我学lca可真的路途曲折,在qbxt,lcy dalao 的vector存图让我这个没接触过stl的蒟蒻完全懵圈,夏令营没有讲,学校里学长讲我又完美错过,心里的痛说不出,,,
还好又找学长单独开了下小灶,额,跑偏了,回归正题。
最近公共祖先的话,貌似挺有用,因为之前多次看到这个字眼(毕竟我只是刚刚打过模板的蒟蒻,题的话还没有去接触),其实我都快放弃lca了,但是学长突然留了一道题需要用lca,,,虽然我还是没有打出来。。。有跑偏了
最近公共祖先,首先我们可以很容易想到,先从一个点往上找它的祖先节点,然后标记下,再找另外一个点的祖先节点,找到的第一个标记的点就是两点的最近公共祖先,但当树退化成链时,复杂度会变为O(N),这样的话,显然是不行的,所以我们在这个基础上利用倍增的思想优化一下,一个一个的跳,有点慢,那我们就两个两个的跳,在此之前先将两个点提到同一深度,然后用倍增的思想,同时往上跳2^n步,这样的复杂度可以优化到O(logn)。
总结一下倍增求lca就以下几点:
1.先确定两个节点的高低,即先假设一个节点在下面,如果不是交换两个节点,使假设的节点确实下面;
2.将两点提至同一深度;
3.两点同时往上跳,直至跳到最近公共祖先;

下面是luogu的模板题 附代码+注释

题目描述

如题,给定一棵有根多叉树,请求出指定两个点直接最近的公共祖先。

输入输出格式

输入格式:

第一行包含三个正整数N、M、S,分别表示树的结点个数、询问的个数和树根结点的序号。

接下来N-1行每行包含两个正整数x、y,表示x结点和y结点之间有一条直接连接的边(数据保证可以构成树)。

接下来M行每行包含两个正整数a、b,表示询问a结点和b结点的最近公共祖先。

输出格式:

输出包含M行,每行包含一个正整数,依次为每一个询问的结果。

输入输出样例

输入样例#1: 复制

5 5 4
3 1
2 4
5 1
1 4
2 4
3 2
3 5
1 2
4 5
输出样例#1: 复制
4
4
1
4
4

说明

时空限制:1000ms,128M

数据规模:

对于30%的数据:N<=10,M<=10

对于70%的数据:N<=10000,M<=10000

对于100%的数据:N<=500000,M<=500000

样例说明:

该树结构如下:

第一次询问:2、4的最近公共祖先,故为4。

第二次询问:3、2的最近公共祖先,故为4。

第三次询问:3、5的最近公共祖先,故为1。

第四次询问:1、2的最近公共祖先,故为4。

第五次询问:4、5的最近公共祖先,故为4。

故输出依次为4、4、1、4、4。


#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cstdlib>using namespace std;
int n;
int m;
int s;
int x,y;
int a,b;
int p;const int maxn = 5211314;
const int inf = 500521;int head[maxn];
int f[inf][20];//f[a][b]意思为a点往上面跳2^b步到达的祖先节点
int deep[maxn];//用来存点的深度
struct edge{//结构体存边int from;int to;int next;
}e[maxn];
inline int read(){//读入优化int x = 0;int f = 1;char c = getchar();while(c<'0'||c>'9'){if(c=='-')f = -f;c = getchar();}while(c<='9'&&c>='0'){x = x*10+c-'0';c= getchar();}return x*f;
}void add(int a,int b){//邻接表存边p++;e[p].from = head[a];e[p].to = b;head[a] = p;
}
void dfs(int root){//dfs深搜建树for(int i = head[root]; i ; i = e[i].from){if(!deep[e[i].to]){deep[e[i].to] = deep[root] + 1;f[e[i].to][0] = root;dfs(e[i].to);}}
}
int lca(int x,int y){if(deep[x] < deep[y]){//确定两点深度x^=y^=x^=y;//异或操作交换x,y两点的值}
//将两点提至同一深度for(int i = 19; i>=0; i--){//这里的19根据数据范围而定if(deep[f[x][i]]>=deep[y]){//这点很重要deep值越大说明深度值越大在下面,x = f[x][i];//我一开始想不明白的原因就是没想通这里}}if(x == y)return x;//如果y是x的祖先节点或者x是y的祖先节点for(int i = 10; i>=0; i--){//让两点同时往上跳if(f[x][i] != f[y][i]){//如果相等说明已经是或者在最近公共祖先之上了x = f[x][i];//两点不断接近最近公共祖先y = f[y][i];//最终一定能到达最近公共祖先的儿子节点}}return f[x][0];//因为是儿子节点所以需要在往上跳一步
}
int main(){n = read();m = read();s = read();for(int i = 1; i<=n-1; i++){x = read();y = read();add(x,y);add(y,x);}deep[s] = 1;f[s][0] = s;dfs(s);for(int i = 1; i<=19; i++){for(int j = 1; j<=n; j++){f[j][i] = f[f[j][i-1]][i-1];}}for(int i = 1; i<=m; i++){a = read();b = read();printf("%d\n",lca(a,b));}return 0;
}

转载于:https://www.cnblogs.com/Euplectella/p/9907297.html

luogu P3379 【模板】最近公共祖先(LCA)相关推荐

  1. POJ 1330 Nearest Common Ancestors / UVALive 2525 Nearest Common Ancestors (最近公共祖先LCA)...

    POJ 1330 Nearest Common Ancestors / UVALive 2525 Nearest Common Ancestors (最近公共祖先LCA) Description A ...

  2. 【C++】最近公共祖先 LCA

    最近公共祖先 百科名片 简单引入 LCA的算法 暴力枚举法 Tarjan离线算法 倍增算法 例题: 题目描述 输入描述 输出描述 样例输入 样例输出 代码 百科名片 最近公共祖先 Lowest Com ...

  3. 最近公共祖先 (LCA) [No. 21]

    问题: 给定一个二叉树,找到两个节点NA, NB的最近公共祖先(LCA). 比如对于下图,4 和 7 的 LCA 是6, 1和13的LCA 是 8. 我们这里先考虑一般的二叉树(BT),然后再考虑这个 ...

  4. 树上倍增法求最近公共祖先LCA

    LCA,最近公共祖先,这个东西有很多作用,因此,如何高效求出LCA就成了一个热点的讨论话题. 下面所有的讨论都以图中这棵树为例子. 先来了解下什么是倍增吧,倍增其实就是二分的逆向,二分是逐渐缩小范围, ...

  5. 图论--最近公共祖先LCA

    最近公共祖先LCA LCA(Least Common Ancestors),即最近公共祖先,是指这样一个问题:在有根树中,找出某两个结点u和v最近的公共祖先(另一种说法,离树根最远的公共祖先) 最近公 ...

  6. 最近公共祖先LCA 【专题@AbandonZHANG】

    参考: <算法艺术与信息学竞赛> --- 刘汝佳 博客:http://blog.163.com/kevinlee_2010/blog/static/16982082020120794613 ...

  7. 算法详解之最近公共祖先(LCA)

    若图片出锅请转至here 概念 首先是最近公共祖先的概念(什么是最近公共祖先?): 在一棵没有环的树上,每个节点肯定有其父亲节点和祖先节点,而最近公共祖先,就是两个节点在这棵树上深度最大的公共的祖先节 ...

  8. 最近公共祖先 LCA Tarjan算法

    来自:http://www.cnblogs.com/ylfdrib/archive/2010/11/03/1867901.html 对于一棵有根树,就会有父亲结点,祖先结点,当然最近公共祖先就是这两个 ...

  9. POJ 1470 Closest Common Ancestors (最近公共祖先LCA 的离线算法Tarjan)

    Tarjan算法的详细介绍,请戳: http://www.cnblogs.com/chenxiwenruo/p/3529533.html #include <iostream> #incl ...

  10. 两节点的最小公共祖先LCA

    一.二叉搜索树中两节点的最小公共祖先: 最初级的题目,在一颗二叉搜索树中寻找两节点的最小公共祖先.根据二叉搜索树的特征,从根节点开始查找,若两节点的val值都小于当前节点,则他们的最小公共祖先就去左子 ...

最新文章

  1. 谷歌大脑发布神经网络的「核磁共振」,并公开相关代码
  2. 8大原因告诉你为什么要使用UPS电源和EPS电源,电气人都知道!
  3. Cocos本地存储LocalStorage
  4. 纯html css博客,纯HTML+CSS打造动画
  5. mysql什么情况会加意向锁_Mysql 数据库意向锁意义
  6. 有哪些适合大学生浏览的网站?
  7. UVALive 4764 dp
  8. 【推荐实践】信息流推荐在凤凰新闻的业务实践
  9. ubuntu17.04新安装之后的软件准备
  10. 电商后台项目 + 源码
  11. Jvisualvm监控远程SpringBoot项目
  12. MIPS单周期CPU设计——lw和sw指令的设计
  13. kindeditor用法简介
  14. visio如何改变折线箭头拐弯方向
  15. Windows的hosts文件在哪里?
  16. oracle 11 ora 12514,客户端连接oracle11出现提示ORA-12514:错误解决方法
  17. 一文带小白玩转NFC、门禁卡
  18. Python操作SQL server向其中插入数据时报错,且报错内容没有指向性。
  19. 虚幻4渲染编程(光线追踪篇)【第一卷:光线追踪篇开篇综述】
  20. php 卡路里计算,那些每天计算卡路里的人,为什么永远也瘦不下来?

热门文章

  1. MFC的exe启动时提示应用程序配置不正确,应用程序未能启动错误
  2. python 实例对象 浅拷贝_Python3 入门教程——浅拷贝深拷贝差别
  3. ASP.NET中String.IndexOf 方法的使用
  4. nginx+keepalived双机热备
  5. oracle 取前10条记录
  6. According to TLD or attribute directive in tag file, attribute value does not accept any expressions
  7. 2057. [ZLXOI2015]殉国
  8. 无限循环的ViewPager
  9. 删除联想OEM分区的方法 推荐!
  10. springboot设置静态资源不拦截的方法