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

根据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. web CSS3 实现3D动态翻牌效果

    使用纯CSS3 实现翻牌效果  需要注意要给子盒子使用绝对定位,这样两个盒子可以完全重合在一起,需要给父盒子一个 transform-style: preserve-3d;让子盒子翻转时保持3D效果, ...

  2. 数据结构中有关顺序表的问题:为何判断插入位置是否合法时if语句中用length+1,而移动元素的for语句中只用length?

    bool ListInsert(SqList &L,int i, ElemType e){ if(i<||i>L.length+) //判断i的范围是否有效 return fals ...

  3. .NET 使用sock5做代理(不是搭建服务端)

    在日常开发中经常会遇到这些需求,爬取数据,都知道现在通常用python爬取是很方便的,但是免不了还有很多伙伴在用NET来爬取,在爬取数据的时候我们知道需要使用代理服务器,如果不用代理,你的IP很有可能 ...

  4. Elasticsearch源码解析:环境搭建

    在之前学习Redis的过程中,我就是一边通过文档/视频学习,一边阅读源码.我发现两种方法可以相辅相成,互相补充.阅读文档可以帮助我们快速了解某个功能,阅读源码有助于我们更深入的理解这一功能的实现及思想 ...

  5. 前端学习(九):CSS基础

    进击のpython ***** 前端学习--CSS基础 CSS的样式可以写在哪呢?其实CSS的样式插入式十分灵活的 按照插入的形势来看,可以分为三种情况 而接下来就对这三种情况进行简单的讨论 内嵌式 ...

  6. Alink漫谈(十四) :多层感知机 之 总体架构

    Alink漫谈(十四) :多层感知机 之 总体架构 目录 Alink漫谈(十四) :多层感知机 之 总体架构 0x00 摘要 0x01 背景概念 1.1 前馈神经网络 1.2 反向传播 1.3 代价函 ...

  7. office 2016激活方法windows

    安装office 2016 win+R,输入“cmd” 根据系统位数和安装的office位数选择相应的命令 OFFICE 64位 和 WINDOWS 64位 cscript "C:\Prog ...

  8. 如何在sed中使用变量,两种方法

    第一 在sed条件中是不认识变量取值的 sed '/$x/d' test 所以要想它能够识别变量 sed "/$x/d/" test 方法简单就是把"单引号"变 ...

  9. Ansible部署zabbix-agent

    playbook目录 zabbix/ ├── hosts ##定义的主机列表 ├── install_zabbix_agent.yml ##安装入口文件 └── roles ├── install_z ...

  10. JAVA基础(jdk安装和环境变量的配置)

    JAVA 1.何为JAVA Java的发展可以归纳如下的几个阶段. (1)第一阶段(完善期):JDK 1.0 ( 1995年推出)一JDK 1.2 (1998年推出,Java更名为Java 2): ( ...