求lca的方法大体有三种:

1.dfs+RMQ(线段树 ST表什么的) 在线

2.倍增 在线

3.tarjan 离线

ps:离线:所有查询全输入后一次解决

在线:有一个查询输出一次

以下模板题为 洛谷 P3379 【模板】最近公共祖先(LCA)

1.首先dfs求出

1>dfs遍历时经过的所有节点的位置

2>每个节点第一次出现的位置

3>每个节点的深度

查询时先取出两个节点的位置求出这两个位置间的深度最小的节点

这个节点就是lca

code:

//By Menteur_Hxy 2068ms
#include<cstdio>
#include<iostream>
#include<cstring>
using namespace std; const int INF=0x3f3f3f3f;
const int MAX=500110; int n,qu,root,cnt;
int head[MAX],ver[MAX*2],first[MAX],log[MAX*2],deep[MAX],f[MAX*2][21]; struct edges{
int to,next;
}edge[MAX*2+5]; void add(int x,int y) {
edge[++cnt].next=head[x];
edge[cnt].to=y;
head[x]=cnt;
} void dfs(int u,int pre) {
ver[++cnt]=u;
first[u]=cnt;
for(int i=head[u];i;i=edge[i].next) {
int v=edge[i].to;
if(v!=pre) {
deep[v]=deep[u]+1;
dfs(v,u);
ver[++cnt]=u;
}
}
} void ST() {
for(int i=1;i<=cnt;i++) f[i][0]=ver[i];
log[1]=0,log[2]=1;
for(int i=3;i<=cnt;i++) {
log[i]=log[i-1];
if(1<<log[i-1]+1==i) log[i]++;
}
int k=log[cnt];
for(int j=1;j<=k;j++)
for(int i=1;i+(1<<j)-1<=cnt;i++) {//一定注意位运算要加括号!!!
if(deep[f[i][j-1]]<deep[f[i+(1<<j-1)][j-1]])
f[i][j]=f[i][j-1];
else f[i][j]=f[i+(1<<j-1)][j-1];
} } int RMQ(int l,int r) {
int k=log[r-l+1];
// cout<<l<<":"<<deep[f[l][k]]<<" "<<r<<":"<<deep[f[r-(1<<k)+1][k]]<<endl;
if(deep[f[l][k]]<deep[f[r-(1<<k)+1][k]])
return f[l][k];
else return f[r-(1<<k)+1][k];
// return min(f[l][k],f[r-1<<k+1][k]);
} int ask(int x,int y) {
x=first[x];y=first[y];
if(x>y) swap(x,y);
return RMQ(x,y);
} int main() {
scanf("%d %d %d",&n,&qu,&root);
for(int i=1;i<n;i++) {
int a,b;
scanf("%d %d",&a,&b);
add(a,b);
add(b,a);
}
cnt=0;
dfs(root,-1);
// for(int i=1;i<=cnt;i++) cout<<ver[i]<<" ";
// cout<<endl;
ST();
for(int i=1;i<=qu;i++) {
int a,b;
scanf("%d %d",&a,&b);
printf("%d\n",ask(a,b));
}
return 0;
}

2.倍增

同样需要dfs,不过只需求出深度和每个节点的父亲

现在设f[i][j] 表示i的第2^j个祖先 (eg:f[i][0] 即为i的父亲)

所以显然有 f[i][j]=f[f[i][j-1]][j-1] (i的第2^j-1个祖先的第2^j-1个祖先是i的2^j个祖先)

查询两节点时,先将深度较大的节点向上移动直到与另一个节点深度相同,判断此时两节点是否相同如果不同,就从大到小枚举尝试往上跳直到两节点父亲相同此时父亲就是lca

code:

//By Menteur_Hxy 1860ms
#include<cstdio>
#include<iostream>
#include<cstdlib>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<ctime>
using namespace std; const int MAX=500010; int n,qu,root,cnt;
int head[MAX],deep[MAX],f[MAX][20];
/*
f[i][j] i的第2^j个祖先
*/
struct edges{
int to,next;
}edge[MAX*2]; void add(int x,int y) {
edge[++cnt].to=y;
edge[cnt].next=head[x];
head[x]=cnt;
} void dfs_bz(int cur) {
for(int i=head[cur];i;i=edge[i].next) {
int v=edge[i].to;
if(v!=f[cur][0]) { //判断!!!
// cout<<cur<<":"<<v<<endl;
f[v][0]=cur;
deep[v]=deep[cur]+1;
dfs_bz(v);
}
}
} void init() {
for(int j=1;(1<<j)<=n;j++)
for(int i=1;i<=n;i++)
if(f[i][j-1]!=-1)
f[i][j]=f[f[i][j-1]][j-1];
} int lca_bz(int x,int y) {
if(deep[x]<deep[y]) swap(x,y);
int d=deep[x]-deep[y];
for(int i=0;d;i++,d>>=1)
if(d&1) x=f[x][i]; // for(int i=0;(1<<i)<=d;i++)
// if((1<<i)&d) x=f[x][i]; 这样写也行 // cout<<x<<","<<y<<endl;
// cout<<deep[x]<<"-"<<deep[y]<<endl;
if(x!=y) {
for(int i=17;i>=0;i--) { //=0也算!!!
if(f[x][i]!=f[y][i])
x=f[x][i],y=f[y][i];
}
return f[x][0];
}
else return x;
} int main(){
scanf("%d %d %d",&n,&qu,&root);
for(int i=1;i<n;i++) {
int a,b;
scanf("%d %d",&a,&b);
add(a,b);
add(b,a);
}
f[root][0]=-1;
dfs_bz(root);
init();
for(int i=1;i<=qu;i++) {
int x,y;
scanf("%d %d",&x,&y);
printf("%d\n",lca_bz(x,y));
}
return 0;
}

3.tarjan

先记录查询的问题

进行dfs,每次在节点(u)返回前查询与它构成问题的所有节点(v),如果其中有之前已经遍历过的节点(vis[v]=true),则这两个节点的lca为find(v),所有v都查过后,将它与它的父节点的集合合并,之后返回。

code;

//By Menteur_Hxy 912ms
#include<cstdio>
#include<iostream>
#include<cstring>
using namespace std; const int MAX=500010; int n,qu,root,cnt;
int head[MAX],f[MAX],head_[MAX],vis[MAX],ans[MAX]; struct edges{
int to,next;
}edge[MAX*2],edge_[MAX*2]; void add(int x,int y) {
edge[++cnt].to=y;
edge[cnt].next=head[x];
head[x]=cnt;
} void add_(int x,int y) {
edge_[++cnt].to=y;
edge_[cnt].next=head_[x];
head_[x]=cnt;
} int find(int x) {
return f[x]==x?x:f[x]=find(f[x]);
} void tarjan(int u,int pre) {
vis[u]=1;
for(int i=head[u];i;i=edge[i].next) {
int v=edge[i].to;
if(v!=pre) {
tarjan(v,u);
int a=find(v),b=find(u);
f[a]=b;
}
}
for(int i=head_[u];i;i=edge_[i].next) {
int v=edge_[i].to;
if(vis[v])
ans[(i+1)/2]=find(v);
}
} int main() {
scanf("%d %d %d",&n,&qu,&root);
for(int i=1;i<n;i++) {
f[i]=i;
int a,b;
scanf("%d %d",&a,&b);
add(a,b);
add(b,a);
}
f[n]=n;
cnt=0;
for(int i=1;i<=qu;i++) {
int a,b;
scanf("%d %d",&a,&b);
add_(a,b);
add_(b,a);
}
tarjan(root,-1);
for(int i=1;i<=qu;i++)
printf("%d\n",ans[i]);
return 0;
}

LCA 最近公共祖先 (笔记、模板)的更多相关文章

  1. lca最近公共祖先(模板)

    洛谷上的lca模板题--传送门 学了求lca的tarjan算法(离线),在洛谷上做模板题,结果后三个点超时. 又把询问改成链式前向星,才ok. 这个博客,tarjan分析的很详细. 附代码-- #in ...

  2. LCA最近公共祖先——Tarjan模板

    LCA(Lowest Common Ancestors),即最近公共祖先,是指在有根树中,找出某两个结点u和v最近的公共祖先. Tarjan是一种离线算法,时间复杂度O(n+Q),Q表示询问次数,其中 ...

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

    #include <iostream> #include <stdio.h> #include <cstring> #include <vector> ...

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

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

  5. lca 最近公共祖先

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

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

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

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

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

  8. LCA近期公共祖先

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

  9. LCA 近期公共祖先 小结

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

随机推荐

  1. 心急的C小加 贪心算法

    心急的C小加 时间限制:1000 ms  |  内存限制:65535 KB 难度:4   描述 C小加有一些木棒,它们的长度和质量都已经知道,需要一个机器处理这些木棒,机器开启的时候需要耗费一个单位的 ...

  2. 1103 N的倍数

    1103 N的倍数  题目来源: Ural 1302 基准时间限制:1 秒 空间限制:131072 KB 分值: 40 难度:4级算法题  收藏  关注 一个长度为N的数组A,从A中选出若干个数,使得 ...

  3. Linux内核中网络数据包的接收-第二部分 select/poll/epoll

    和前面文章的第一部分一样,这些文字是为了帮别人或者自己理清思路的.而不是所谓的源代码分析.想分析源代码的,还是直接debug源代码最好,看不论什么文档以及书都是下策. 因此这类帮人理清思路的文章尽可能 ...

  4. C语言读取文件大量数据到数组

    针对.txt文档的大量有规律数据,譬如100行8列的数据将其读取到二维数组(矩阵)中,留作之后的数据处理. 改程序通过宏定义的方法来确定将要读取程序的行数和列数,将数据读取到二维数组data[100] ...

  5. ADO.NET Entity Framework Extensions

    一.情景 如果你的项目中有返回多结果集的存储过程. 如果你的项目要和老项目中的ADO.Net共用事务. 如果你要动态的创建数据库的表. 但是你还是希望使用Entity Framework.那么继续往下 ...

  6. 深度学习实战篇-基于RNN的中文分词探索

    深度学习实战篇-基于RNN的中文分词探索 近年来,深度学习在人工智能的多个领域取得了显著成绩.微软使用的152层深度神经网络在ImageNet的比赛上斩获多项第一,同时在图像识别中超过了人类的识别水平 ...

  7. WebP 文件及其编码解码工具(WebPconv)

    1. webp 文件 与JPEG相同,WebP 是一种有损压缩利用预测编码技术. WebP 是 Google 新推出的影像技术,它可让网页图档有效进行压缩,同时在质量相同的情况下,WebP 格式图像的 ...

  8. 【转】坑爹的AsyncTask之根本停不下来

    原文网址:http://www.jianshu.com/p/0c6f4b6ed558 上篇<坑爹的AsyncTask之内存泄露>已经简单的探讨过线程使用不当会造成内存泄露的问题,在Acti ...

  9. angular2 使用swiper

    欢迎加入前端交流群交流知识&&获取视频资料:749539640 第一步: npm install swiper --save 第二步:下载swiper ts支持(http://micr ...

  10. 46.Qt 使用OpenGL绘制立方体

    main.cpp #include <QApplication> #include <iostream> #include "vowelcube.h" in ...