题目链接

一、 时间戳法(本质上是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. mysql是如何实现mvcc的

    mvcc的概念 mvcc即多版本并发控制,是一种并发控制的策略,能让数据库在高并发下做到安全高效的读写,提升数据库的并发性能; 是一种用来解决并发下读写冲突的无锁解决方案,为事务分配单向增长时间戳,为 ...

  2. 《Design by Contract for Embedded Software》 翻译

    原文: Design by Contract for Embedded Software (state-machine.com) Design by Contract is the single mo ...

  3. 关于Module Not Found Error No module named Crypto解决

    前言 之前就遇到这个问题, 当然是windows上具有的问题 问题描述 from Crypto.Cipher import AES 出现 ModuleNotFoundError: No module ...

  4. sublime text配置java运行环境

    java正常运行流程 首先一个 test.java 文件 如果要运行这个文件需要先编译 javac test.java 然后在同级目录会生成一个test.class 文件 然后 java test 来 ...

  5. postman的运用

    链接: https://pan.baidu.com/s/1gfaKoAv 密码: dp7t 最近要测试和其他系统对接,忽然想起了postman这款url测试神器. 现分享如下: 下载完成后解压到文件夹 ...

  6. React Server Component: 混合式渲染

    作者:谢奇璇 React 官方对 Server Comopnent 是这样介绍的: zero-bundle-size React Server Components. 这是一种实验性探索,但相信该探索 ...

  7. 【每日一题】【直接循环&二分查找】2022年2月10日-NC32 求平方根

    描述实现函数 int sqrt(int x).计算并返回 x 的平方根(向下取整) 方法1:直接循环 import java.util.*; public class Solution { /** * ...

  8. python爬虫爬取网易云音乐(超详细教程,附源码)

    一. 前言 先说结论,目前无法下载无损音乐,也无法下载vip音乐. 此代码模拟web网页js加密的过程,向api接口发送参数并获取数据,仅供参考学习,如果需要下载网易云音乐,不如直接在客户端下载,客户 ...

  9. docker-compose + mysql8.x 主从数据库配置

    0.准备 (略过docker的安装与镜像拉取) docker / docker-compose 安装 拉取 mysql 8.x 1. master和slave的mysql配置 master: [mys ...

  10. 一文聊透Apache Hudi的索引设计与应用

    Hudi索引在数据读和写的过程中都有应用.读的过程主要是查询引擎利用MetaDataTable使用索引进行Data Skipping以提高查找速度;写的过程主要应用在upsert写上,即利用索引查找该 ...