LCA (Tarjan&倍增)
LCA_Tarjan
参考博客:https://www.cnblogs.com/JVxie/p/4854719.html
LCA的Tarjan写法需要结合并查集
从叶子节点往上并
int Find (int x) {
return x == pre[x] ? x:pre[x] = Find(pre[x]);
}
void dfs(int x,int w,int fa) {
d[x] = w; //点x的深度
//遍历与点x相连的点(除了已经访问过的和其父节点)
for (int i = ; i < v[x].size(); i++) {
if (!vis[v[x][i]] && v[x][i] != fa) {
dfs(v[x][i],w+,x);
pre[v[x][i]] = x;
vis[v[x][i]] = ;
}
}
//访问所有和x有询问关系的e
//如果e被访问过;
//x,e的最近公共祖先为find(e);
}
LCA_倍增
参考博客:https://www.cnblogs.com/zhouzhendong/p/7256007.html
先写暴力写法:当x,y深度不同时先把深的调到和浅的同一深度再一起往前面跳找其最近公共祖先。
void dfs(int f,int u){
fa[u]=f;
d[u]=d[f]+;
for (int i=;i<v[u].size();i++)
if (v[u][i]!=f) dfs(u,v[u][i]);
}
int LCA(int a,int b){
if (d[a]>d[b])
swap(a,b);
while (d[b]>d[a]) b=fa[b];
while (a!=b) a=fa[a],b=fa[b];
return a;
}
显然,一个一个的跳太慢了,我们可以考虑通过二进制进行优化,也就是倍增。
void dfs(int f,int u) {
fa[u][] = f;
dep[u] = dep[f] + ;
for (int i = ; i <= ; i++) {
fa[u][i] = fa[fa[u][i-]][i-];
//其他操作
}
for (int i = ; i < v[u].size(); i++) {
if (f == v[u][i]) continue;
dfs(u,v[u][i]);
}
}
void lca(int x,int y) {
memset(ans,,sizeof(ans));
if (dep[x] < dep[y]) swap(x,y);
for (int i = ; i >= ; i--)
if (dep[fa[x][i]] >= dep[y]) {
//其他操作
x = fa[x][i];
}
if (x == y) {
return;
}
for (int i = ; i >= ; i--)
if (fa[x][i] != fa[y][i]) {
//其他操作
x = fa[x][i],y = fa[y][i];
}
}
例题:洛谷P3292 [SCOI2016]幸运数字 | 题解
LCA (Tarjan&倍增)的更多相关文章
- [codevs1036]商务旅行<LCA:tarjan&倍增>
题目链接:http://codevs.cn/problem/1036/ 今天翻箱倒柜的把这题翻出来做了,以前做的时候没怎么理解,所以今天来重做一下 这题是一个LCA裸题,基本上就把另一道裸题小机房的树 ...
- HDU 2874 Connections between cities(LCA Tarjan)
Connections between cities [题目链接]Connections between cities [题目类型]LCA Tarjan &题意: 输入一个森林,总节点不超过N ...
- POJ 1986 Distance Queries(LCA Tarjan法)
Distance Queries [题目链接]Distance Queries [题目类型]LCA Tarjan法 &题意: 输入n和m,表示n个点m条边,下面m行是边的信息,两端点和权,后面 ...
- LCA Tarjan方法
LCA Tarjan方法 不得不说,高中生好厉害,OI大佬,感觉上个大学好憋屈啊! 说多了都是眼泪 链接拿去:http://www.cnblogs.com/JVxie/p/4854719.html
- 关于LCA的倍增解法的笔记
emmmmm近日刚刚学习了LCA的倍增做法,写一篇BLOG来加强一下印象w 首先 何为LCA? LCA“光辉”是印度斯坦航空公司(HAL)为满足印度空军需要研制的单座单发轻型全天候超音速战斗攻击机,主 ...
- LCA tarjan+并查集POJ1470
LCA tarjan+并查集POJ1470 https://www.cnblogs.com/JVxie/p/4854719.html 不错的一篇博客啊,让我觉得LCA这么高大上的算法不是很难啊,嘻嘻嘻 ...
- LCA的倍增算法
LCA,即树上两点之间的公共祖先,求这样一个公共祖先有很多种方法: 暴力向上:O(n) 每次将深度大的点往上移动,直至二者相遇 树剖:O(logn) 在O(2n)预处理重链之后,每次就将深度大的沿重链 ...
- [模板]LCA的倍增求法解析
题目描述 如题,给定一棵有根多叉树,请求出指定两个点直接最近的公共祖先. 输入输出格式 输入格式: 第一行包含三个正整数N.M.S,分别表示树的结点个数.询问的个数和树根结点的序号. 接下来N-1行每 ...
- hihoCoder #1067 : 最近公共祖先·二 [ 离线LCA tarjan ]
传送门: #1067 : 最近公共祖先·二 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 上上回说到,小Hi和小Ho用非常拙劣——或者说粗糙的手段山寨出了一个神奇的网站 ...
- LCA算法解析-Tarjan&倍增&RMQ
原文链接http://www.cnblogs.com/zhouzhendong/p/7256007.html UPD(2018-5-13) : 细节修改以及使用了Latex代码,公式更加美观.改的过程 ...
随机推荐
- CODE FESTIVAL 2017 qual B C 3 Steps(补题)
总感觉这题是个题意杀,理解错题目了,看了好久才发现题目意思:操作是让,只要两点没有直接相连,而且只要有一条路的距离3,就可以把这两点连接起来. 按照题解中讲的,可以把图分为二分图和非二分图来解.不过题 ...
- data-属性的作用
data-用于存储页面或应用程序的私有自定义数据,赋予我们在所有HTML元素上嵌入自定义data属性的能力,存储的数据能被页面的JS利用,以创建更好的用户体验. <div id="bo ...
- 在 Jenkins Windows Agent 节点上执行 Shell 命令
Jenkins 在 Windows agent 上执行shell 命令,听起来很有意思,以下方法可以在 Jenkins 中执行一些简单的 shell 脚本,如果是复杂脚本就交给 Linux agent ...
- Python--day30--基于tcp协议的套接字socket
socket 一开始被设计用在一台主机上多个应用程序之间通信. 是应用层与TCP/IP协议族通信的中间软件抽象层,它是一组接口. 是一个模块,是ip+port,门面模式是一种设计模式. socket通 ...
- Element节点输出到System.out
protected void writeElementToFile(Element valrespEle) { try { TransformerFactory transformerFactory ...
- Vue 组件中的data数据
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- H3C配置路由器作为TFTP客户端
- Makefile记录
需要把sum.c编译汇编成可执行程序zzj zzj:sum.o gcc -o zzj sum.osum.o:sum.c gcc -c -o sum.o sum.cclean: rm -rf *.o z ...
- shell截取字符串的8种方法
参考文献: linux中shell截取字符串方法总结 [Linux]如何在Shell脚本中计算字符串长度? 截取字符串的方法一共有八种,主要为以下方法 shell中截取字符串的方法有很多中, ${ex ...
- spring boot 多数据源加载原理
git代码:https://gitee.com/wwj912790488/multiple-data-sources DynamicDataSourceAspect切面 必须定义@Order(-10) ...