高级的算法——倍增!!!

根据LCA的定义,我们可以知道假如有两个节点x和y,则LCA(x,y)是 x 到根的路 径与 y 到根的路径的交汇点,同时也是 x 和 y 之间所有路径中深度最小的节 点,所以,我们可以用遍历路径的方法求 LCA。

但想想都知道啦,这种遍历的方法肯定too slow,最坏情况时可达到O(n),数据大点儿,就光荣TLE了。

所以我们高级的化身——倍增算法就出现了!

谈谈倍增——

倍增简单来讲就是两个点跳到同一高度后,再一起往上跳,直到跳到一个共同的点,就能找到它们的最近公共祖先啦

具体来说,分为以下几个步骤——

首先,我们得找几个帮手——

数组 解释
f[][] 预处理倍增f[v][i]          
d[][] 记录节点深度
vis[][] 判重,以防搜到节点的爸爸
head[][] 存图时用der~(不懂
dis[][]

一个节点到根的最短距离

(有权值记得加权值)

Step1:  预处理每个结点的深度和该节点的父亲节点

我们用 d 数组来表示每个结点的深度,设节点 1 为根,d[1]=0,初始化 f[v][0]=u, 表示 v 的 20 的祖先节点为 u。

tips:如果树是无根树时需要把它变成有根数(随便找个点就OK)

Step2:  预处理倍增f[v][i],2的祖先也是该点 2j-1 祖先的祖先 

核心: f[u][j] = f [f [u][j-1]][j-1] 

不懂的盆友,我们来举个例子——

我们有这样的一张图,我们对它进行预处理就会变成这样——

*图源

还不懂的,就手动模拟一波~

Step3:  查询时,深度大的先往上跳,直至与深度小的点再同一层。若此时两节点相同直接返回此节点(在同一条链上),如果不同,就利用倍增的思想,同时让 x 和 y 向上找,直到找到深度相等且最小的 x 和 y 的祖先 x′,y′,满足 x′≠y′。此时他们的父结点即为 x 和 y 的最近公共祖先 LCA。 

倍增解法是 LCA 问题的在线解法,整体时间复杂度是 O(VlogV+QlogV),其 中 Q 是总查询次数。

看起来是不是还不错?那我们来看道题吧~

谈谈题目——

Description

给定一棵n个点的树,Q个询问,每次询问点x到点y两点之间的距离。

Input

第一行一个正整数n,表示这棵树有n个节点;接下来n-1行,每行两个整数x,y表示x,y之间有一条连边;然后一个整数Q,表示有Q个询问;接下来Q行每行两个整数x,y表示询问x到y的距离。

对于全部数据,1≤n≤105,1≤x,y≤n。

Output

输出Q行,每行表示每次询问的答案。

Example

stdin 

6

1 2

1 3

2 4

2 5

3 6

2

2 6

5 6

stdout 

3

4

*原题

Thinking 

说实话这题没什么好讲的叭,就是求LCA就OK啦,板子题耶!

上代码——

 #include<bits/stdc++.h>
using namespace std;
int t,q,tot;
int dis[];
int vis[];
struct node{
int nxt,to;
}e[];
int head[];
int f[][];
int d[]; void add(int u, int v){
e[++tot].to = v;
e[tot].nxt=head[u];
head[u] = tot;
} void dfs(int u, int fa, int dep){ //预处理深度,倍增数组
vis[u] = ;
f[u][] = fa; //初始化记录父节点
d[u] = dep;
for(int j = ; j <= ; j++) f[u][j] = f[f[u][j-]][j-];
for(int i = head[u]; i; i = e[i].nxt){
int v = e[i].to;
if(vis[v]) continue;
dfs(v,u,dep+);
}
} int lca(int x, int y){ //倍增求 LCA
if(d[x] > d[y]) swap(x,y);
for(int i = ; i >= ; i--){
if(d[f[y][i]] >= d[x]) y = f[y][i]; //深度较深节点先往上跳至同一深度
if(x == y) return x;
}
for(int i = ; i >= ; i--){
if(f[x][i] != f[y][i]) x = f[x][i], y= f[y][i];
}
return f[x][];
} int main(){
scanf("%d",&t);
tot = ;
for(int i = ; i < t; i++){
int u,v;
scanf("%d%d",&u,&v);
add(u,v),add(v,u);
}
dfs(,,);
cin >> q;
for(int i = ; i <= q; i++){
int x,y;
scanf("%d%d",&x,&y);
printf("%d\n",d[x]+d[y]-*d[lca(x,y)]); //两个节点到跟的距离减去重复计算的它们公共祖先到跟的距离
}
}

解释一下这个东西 d[x] + d[y] - 2 * d[lca(x,y)];

画张图——

d[x] 和 d[y] 就是红色那两条东西

d[lca(x,y)] 就是蓝色那条

d[x] + d[y] - 2*d[lca(x,y)] 就是绿色那条啦~

当然这是没有权值得时候,我们默认深度差不多等于距离,但有了权值就不一样了。

我们再来看一道板子题——

Description

给出n个点的一棵树,多次询问两点之间的最短距离。

注意:边是双向的。

Input

第一行为两个整数n和m。n表示点数,m表示询问次数; 下来n-1行,每行三个整数x,y,k,表示点x和点y之间存在一条边长度为k;在接下来m行,每行两个整数x,y,表示询问点x到点y的最短距离。

对于全部数据,2≤n≤104,1≤m≤2×104,0<k≤100,1≤x,y≤n。

Output 

输出m行。对于每次询问,输出一行。

Example

stdin1

2 2 1 2 100 1 2 2 1

stdout1

100 100

stdin2

3 2 1 2 10 3 1 15 1 2 3 2

stdout2 

10 25

*原题

 #include<bits/stdc++.h>
using namespace std;
int n,m,q,tot;
int dis[];
int vis[];
struct node{
int nxt,to;
int w;
}e[];
int head[];
int f[][];
int d[]; void add(int u, int v, int w){
e[++tot].to = v;
e[tot].w = w;
e[tot].nxt=head[u];
head[u] = tot;
} void dfs(int u, int fa, int dep){
vis[u] = ;
f[u][] = fa;
d[u] = dep;
for(int j = ; j <= ; j++) f[u][j] = f[f[u][j-]][j-];
for(int i = head[u]; i; i = e[i].nxt){
int v = e[i].to;
if(vis[v]) continue;
dis[v] = dis[u] + e[i].w;
dfs(v,u,dep+);
}
} int lca(int x, int y){
if(d[x] > d[y]) swap(x,y);
for(int i = ; i >= ; i--){
if(d[f[y][i]] >= d[x]) y = f[y][i];
if(x == y) return x;
}
for(int i = ; i >= ; i--){
if(f[x][i] != f[y][i]) x = f[x][i], y= f[y][i];
}
return f[x][];
}
int main(){
scanf("%d%d",&n,&m);
for(int i = ; i < n; i++){
int u,v,w;
scanf("%d%d%d",&u,&v,&w);
add(u,v,w);
add(v,u,w);
}
dfs(,,);
for(int i = ; i <= m; i++){
int x,y;
scanf("%d%d",&x,&y);
printf("%d\n",dis[x]+dis[y]-*dis[lca(x,y)]);
}
}

注意到没有?

这一道题是有权值的,所以最后输出的时候输出的是 dis[x] + dis[y] - 2 * dis[lca(x,y)]

总结一下

其实LCA还有别的不同的求法,下次在和你们讲吧(其实是我还没学会)

这次就先到这儿吧~

拜拜~

(如果文章有不对的地方,请指出,谢谢啦^=^)

 

【lhyaaa】最近公共祖先LCA——倍增!!!的更多相关文章

  1. 最近公共祖先 LCA 倍增算法

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

  2. luogu3379 【模板】最近公共祖先(LCA) 倍增法

    题目大意:给定一棵有根多叉树,请求出指定两个点直接最近的公共祖先. 整体步骤:1.使两个点深度相同:2.使两个点相同. 这两个步骤都可用倍增法进行优化.定义每个节点的Elder[i]为该节点的2^k( ...

  3. 最近公共祖先 LCA 倍增法

    [简介] 解决LCA问题的倍增法是一种基于倍增思想的在线算法. [原理] 原理和同样是使用倍增思想的RMQ-ST 算法类似,比较简单,想清楚后很容易实现. 对于每个节点u , ancestors[u] ...

  4. Luogu 2245 星际导航(最小生成树,最近公共祖先LCA,并查集)

    Luogu 2245 星际导航(最小生成树,最近公共祖先LCA,并查集) Description sideman做好了回到Gliese 星球的硬件准备,但是sideman的导航系统还没有完全设计好.为 ...

  5. POJ 1470 Closest Common Ancestors(最近公共祖先 LCA)

    POJ 1470 Closest Common Ancestors(最近公共祖先 LCA) Description Write a program that takes as input a root ...

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

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

  7. [模板] 最近公共祖先/lca

    简介 最近公共祖先 \(lca(a,b)\) 指的是a到根的路径和b到n的路径的深度最大的公共点. 定理. 以 \(r\) 为根的树上的路径 \((a,b) = (r,a) + (r,b) - 2 * ...

  8. suoi31 最近公共祖先2 (倍增lca)

    根为r时x.y的公共祖先,就是lca(x,r),lca(x,y),lca(r,y)中深度最大的那一个,不要再在倍增的时候判来判去还判不对了... #include<bits/stdc++.h&g ...

  9. [luogu3379]最近公共祖先(树上倍增求LCA)

    题意:求最近公共祖先. 解题关键:三种方法,1.st表 2.倍增法 3.tarjan 此次使用倍增模板(最好采用第一种,第二种纯粹是习惯) #include<cstdio> #includ ...

随机推荐

  1. windows python的多进程

    最近打比赛,apply操作极慢,队友使用了线程池,用多核开辟多线程跑,加速. 在阿里平台上,都没问题. 我是win10系统+jupyter notebook 多线程那个模块运行,会显示一直运行,p.c ...

  2. 用python批量处理Excel表格,处理结果又快又好,做办公室最靓的那个仔

    使用python批量处理Excel数据     让你根据Excel上所有人的身份证号码,提取出公司员工的生日 让你每个月都将公司所有人的考勤数据整理一下 类似这样的格式化的重复操作,你还在每次都使用的 ...

  3. Lucas定理 & Catalan Number & 中国剩余定理(CRT)

    又双叒叕来水数论了 今天来学习\(Lucas \:\ \& \:\ Catalan Number\) 两者有着密切的联系(当然还有CRT),所以放在一起学习一下 \(Lucas\) 定义\(\ ...

  4. sqlserver安装出现找不到数据库引擎错误

    sqlserver安装出现找不到数据库引擎错误 问题的解决 第一次安装SQL server,发现它较于Oracle,都有安装卸载十分麻烦的特点.刚开始安装,就让我频繁遇到这个“找不到数据库引擎”的错误 ...

  5. 【Vue组件通信】props、$ref、$emit,组件传值

    1.什么是组件通信 组件间如何通信,也就成为了vue中重点知识,组件通信,涉及到组件之间数据的传递.类似NET POST/GET参数传递. Vue基本的三种传递方式** (props.\(ref.\) ...

  6. 华东师范大学数学分析课本p294,引理3的我的更正证明

    书上的证明是一个特例,我的证明是,如果这个特例不成立,就继续做n-1,直到特例的情况出现,即可.

  7. java基础(十)--空指针异常

    空指针异常即:java.lang.NUllPointException异常,主要用于在对象为null的情况下,调用对象的方法或对象的属性时会抛出. 举例说明: public class TestBas ...

  8. 使用expect在script中切换到root用户(精华)

    使用expect在script中切换到root用户 1.尚观版本 http://www.uplook.cn/biancheng/133/1335040/ 1 a. 命令行: /usr/bin/expe ...

  9. Linux阶段总结

    Linux总结 一.学习心得: 在学习本阶段关于Linux阶段的课程时,让我对Linux有了一个大概的了解. 我了解到Linux操作系统是基于最初的Unix系统而开发出来的: 在学习Linux的时候, ...

  10. 总结HashMap实现原理分析

    一.底层数据结构在JDK1.6,JDK1.7中,HashMap采用位桶+链表实现,即使用链表处理冲突,同一hash值的键值对会被放在同一个位桶里,当桶中元素较多时,通过key值查找的效率较低. 而JD ...