题目链接

一、 时间戳法(本质上是dfs序)

#include<cstdio>
using namespace std;
const int NN = 5e5+8;
int n,m,s;
int tin[NN],tout[NN],timetable;
int fa[NN][30]; struct Edge{
int next,to;
}edge[NN<<1];
int head[NN],cnt;
void init(){
cnt = 1;
for(int i = 1; i <= n; i++)head[i] = -1;
}
void add_edge(int u,int v){
edge[++cnt].next = head[u];
edge[cnt].to = v;
head[u] = cnt;
}//链式前向星 void dfs(int now,int father){
fa[now][0]=father;
tin[now] = ++timetable;
for(int j = 1; j <= 29 ; j++){
fa[now][j] = fa[fa[now][j-1]][j-1];
}
for(int i = head[now]; i != -1; i = edge[i].next){
int tt = edge[i].to;
if(father != tt)dfs(tt,now);
}
tout[now] = ++timetable;
return;
}//一遍dfs预处理fa数组,时间戳 bool isfather(int a,int b){
return tin[a]<tin[b]&&tout[b]<tout[a];
}//可以知道先进入dfs后退出的一定是更先搜到
int LCA(int x,int y){
if(x==y)return x;
if(isfather(x,y))return x;
if(isfather(y,x))return y;
int now = x;
for(int i = 29; i >= 0; i--){
if(!isfather(fa[now][i],y))now = fa[now][i];
}
return fa[now][0];
}
int main(){
scanf("%d%d%d",&n,&m,&s);
init();
for(int i = 1,u,v; i < n; i++){
scanf("%d%d",&u,&v);
add_edge(u,v);
add_edge(v,u);
}
dfs(s,s);
while(m--){
int u,v;
scanf("%d%d",&u,&v);
printf("%d\n",LCA(u,v));
}
return 0;
}

码量:1228

二、 普通法(先跳到同一深度,再向上跳)

#include<cstdio>
#include<algorithm>
using namespace std;
const int NN = 5e5+8;
int n,m,s; int fa[NN][30],dep[NN];
struct Edge{
int next,to;
}edge[NN<<1];
int head[NN],cnt;
void init(){
cnt = 1;
for(int i = 1; i <= n; i++)head[i] = -1;
}
void add_edge(int u,int v){
edge[++cnt].next = head[u];
edge[cnt].to = v;
head[u] = cnt;
}//链式前向星 void dfs(int depth,int now,int father){
fa[now][0] = father;
dep[now] = depth;
for(int i = 1; i <= 29; i++){
fa[now][i] = fa[fa[now][i-1]][i-1];
}
for(int i = head[now]; i != -1; i = edge[i].next){
int tt = edge[i].to;
if(tt != father)dfs(depth+1,tt,now);
}
}//一遍dfs处理fa数组 int LCA(int x,int y){
if(dep[y]>dep[x])swap(x,y);
int t = dep[x]-dep[y];
for(int i = 0; t; i++){
if(t&1)x = fa[x][i];
t>>=1;
}//跳到同一深度
if(x==y)return x;
for(int i = 29; i >= 0; i--){
if(fa[x][i] != fa[y][i])x = fa[x][i],y = fa[y][i];
} //倍增
return fa[x][0];
} int main(){
scanf("%d%d%d",&n,&m,&s);
init();
for(int i = 1,u,v; i < n; i++){
scanf("%d%d",&u,&v);
add_edge(u,v);
add_edge(v,u);
}
dfs(1,s,s);
while(m--){
int u,v;
scanf("%d%d",&u,&v);
printf("%d\n",LCA(u,v));
}
return 0;
}

码量:1179



两种代码自取

【模板】倍增求LCA的更多相关文章

  1. [算法模板]倍增求LCA

    倍增LCA \(fa[a][i]\)代表a的第\(2^{i}\)个祖先. 主体思路是枚举二进制位,让两个查询节点跳到同一高度然后再向上跳相同高度找LCA. int fa[N][21], dep[N]; ...

  2. 倍增求lca模板

    倍增求lca模板 https://www.luogu.org/problem/show?pid=3379 #include<cstdio> #include<iostream> ...

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

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

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

    前几天做faebdc学长出的模拟题,第三题最后要倍增来优化,在学长的讲解下,尝试的学习和编了一下倍增求LCA(我能说我其他方法也大会吗?..) 倍增求LCA: father[i][j]表示节点i往上跳 ...

  5. [算法]树上倍增求LCA

    LCA指的是最近公共祖先(Least Common Ancestors),如下图所示: 4和5的LCA就是2 那怎么求呢?最粗暴的方法就是先dfs一次,处理出每个点的深度 然后把深度更深的那一个点(4 ...

  6. 【倍增】洛谷P3379 倍增求LCA

    题目描述 如题,给定一棵有根多叉树,请求出指定两个点直接最近的公共祖先. 输入输出格式 输入格式: 第一行包含三个正整数N.M.S,分别表示树的结点个数.询问的个数和树根结点的序号. 接下来N-1行每 ...

  7. hdu 2586 How far away ? 倍增求LCA

    倍增求LCA LCA函数返回(u,v)两点的最近公共祖先 #include <bits/stdc++.h> using namespace std; *; struct node { in ...

  8. 【题解】洛谷P4180 [BJWC2010] 严格次小生成树(最小生成树+倍增求LCA)

    洛谷P4180:https://www.luogu.org/problemnew/show/P4180 前言 这可以说是本蒟蒻打过最长的代码了 思路 先求出此图中的最小生成树 权值为tot 我们称这棵 ...

  9. 树链剖分与倍增求LCA

    树链剖分与倍增求\(LCA\) 首先我要吐槽机房的辣基供电情况,我之前写了一上午,马上就要完成的时候突然停电,然后\(GG\)成了送链剖分 其次,我没歧视\(tarjan LCA\) 1.倍增求\(L ...

  10. [学习笔记] 树上倍增求LCA

    倍增这种东西,听起来挺高级,其实功能还没有线段树强大.线段树支持修改.查询,而倍增却不能支持修改,但是代码比线段树简单得多,而且当倍增这种思想被应用到树上时,它的价值就跟坐火箭一样,噌噌噌地往上涨. ...

随机推荐

  1. OpenHarmony移植案例: build lite源码分析之hb命令__entry__.py

    摘要:本文介绍了build lite 轻量级编译构建系统hb命令的源码,主要分析了_\entry__.py文件. 本文分享自华为云社区<移植案例与原理 - build lite源码分析 之 hb ...

  2. ssh明文密码小工具:sshpass

    Xshell 远程时可以用如下方式进行明文密码免交互登陆: ssh 用户:密码@ip 可换到linux时,就不行了,在某些不适合使用公钥免密的情况下,到底该怎样免交互登陆? 于是我找到了sshpass ...

  3. Java:自定义排序与sort()函数

    自定义排序与Arrays.sort() 本篇题目来源:2022/11/13 Leetcode每日一题:https://leetcode.cn/problems/custom-sort-string 给 ...

  4. 在Windows模拟器中使用LVGL8.3

    引言 LVGL是一个跨平台.轻量级.易于移植的图形库.也因其支持大量特性和其易于裁剪,配置开关众多,且版本升级较快,不同版本之间存在一定的差异性,相关的使用教程有一定的滞后性,由于缺少最新版本的中文教 ...

  5. <一>智能指针基础

    代码1 int main(){ //裸指针,手动开辟,需要自己释放,如果忘记了或者因为 //程序逻辑导致p没有释放,那么就会导致内存泄漏 int *p=new int(10); if(***){ re ...

  6. .NET性能优化-ArrayPool同时复用数组和对象

    前两天在微信后台收到了读者的私信,问了一个这样的问题,由于私信回复有字数和篇幅限制,我在这里统一回复一下.读者的问题是这样的: 大佬您好,之前读了您的文章受益匪浅,我们有一个项目经常占用 7-8GB ...

  7. RFN-Nest_ An end-to-end residual fusion network for infrared and visible images 论文解读

    RFN-Nest 2021 研究 图像融合分为三步:特征提取,融合策略,图像重建. 当前端到端的图像融合方法:基于GAN的.还有本文提出的 研究背景:当前设计的融合策略在为特定任务生成融合图像方面是比 ...

  8. JS传值与应用

    问题提出:在进行页面书写的时候,有时候需要进行动态页面拼接,在动态拼接的时候,涉及到函数的调用,函数的传值可能是HTML标签,或者含有json的标签,这样在传值时就有可能出现问题,由于"&q ...

  9. Python 缩进语法的起源:上世纪 60-70 年代的大胆创意!

    上个月,Python 之父 Guido van Rossum 在推特上转发了一篇文章<The Origins of Python>,引起了我的强烈兴趣. 众所周知,Guido 在 1989 ...

  10. get请求拼接数组转字符串

    get请求拼接数组转换成字符串