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

根据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. Django框架01 / http协议、web框架本质

    Django框架01 / http协议.web框架本质 目录 Django框架01 / http协议.web框架本质 1.http协议 1.1 http协议简介 1.2 什么是http协议 1.3 H ...

  2. Sequential Minimal Optimization: A Fast Algorithm for Training Support Vector Machines 论文研读

    摘要 本文提出了一种用于训练支持向量机的新算法:序列最小优化算法(SMO).训练支持向量机需要解决非常大的二 次规划(QP)优化问题.SMO 将这个大的 QP 问题分解为一系列最小的 QP 问题.这些 ...

  3. GPO - Folder Mapping via GPO

    Create a Group Policy on AD DC Server. The GPO policy will come into effect on the next login, or us ...

  4. BT面板安装教程

    面板特点 一键配置服务器环境(LAMP/LNMP) 一键安全重启 一键创建管理网站.ftp.数据库 一键配置(定期备份.数据导入.伪静态.301.SSL.子目录.反向代理.切换PHP版本) 一键安装常 ...

  5. 多国正在遭遇新型勒索病毒Petya侵袭

    北京时间2017年6月27日晚,据外媒消息,多国正在遭遇 Petya 勒索病毒袭击,政府.银行.电力系统.通讯系统.企业以及机场都受到不同程度影响.请予关注,并做相应防范.相关事件描述及防范措施如下: ...

  6. 利用tox打造自动自动化测试框架

    什么是tox tox官方文档的第一句话 standardize testing in Python,意思就是说标准化python中的测试,那是不是很适合测试人员来使用呢,我们来看看他究竟是什么? 根据 ...

  7. 设计模式:abstract factory模式

    含义:抽象工厂将“抽象零件”组装成“抽象产品” 理解:相比于工厂方法模式,可以根据不同的接口创建不同的产品,说白了就是将一个接口变成两个接口,各自返回不同的抽象产品 例子: class Car //抽 ...

  8. go test 测试用例那些事(二) mock

    关于go的单元测试,之前有写过一篇帖子go test测试用例那些事,但是没有说go官方的库mock,很有必要单独说一下这个库,和他的实现原理. mock主要的功能是对接口的模拟,需要在写代码的时候定义 ...

  9. Docker 入门教程(3)——Dockerfile

    Dockerfile Dockerfile是一个文本文件,用来定制镜像. 镜像是分层存储的,前一层会是下一层的基础.而镜像的定制就是定制每一层镜像在上一层做了什么改变. Dockerfile其内包含一 ...

  10. JMS微服务开发示例(一)Hello world

    网关部署 1.在网关服务器上,安装.net core 3.1运行环境: 2.到 https://www.cnblogs.com/IWings/p/13354541.html 下载Gateway.zip ...