浅谈关于LCA
prologue
本身只会 tarjan 和 倍增法求LCA 的,但在发现有一种神奇的\(O(1)\) 查询 lca 的方法,时间优化很明显。
main body
倍增法
先讨论倍增法,倍增法求 lca 是一种很常见普遍的方法,这里直接放代码了,其本身的内核就是让较低点每次都跳 $ 2 ^ k $ 步,如果跳的比另一个高了,就不跳那么高,跳 \(2 ^ {k-1}\) 步,这就用对数级的复杂度求出来了 LCA。
code
比较基础,建议直接背过。
inline void bfs()
{
memset(dep, 0x3f, sizeof dep);
ll hh = 0, tt = -1;
q[++ tt] = 1;
dep[0] = 0, dep[1] = 1;
while(hh <= tt)
{
ll u = q[hh ++ ];
for(rl i = h[u]; ~i; i = ne[i])
{
ll v = e[i];
if(dep[v] > dep[u] + 1)
{
dep[v] = dep[u] + 1;
fa[v][0] = u;
q[++ tt] = v;
for(rl k=1; k <= 20; ++ k)
fa[v][k] = fa[fa[v][k - 1]][k - 1];
}
}
}
}
inline ll lca(ll a, ll b)
{
if(dep[a] < dep[b]) swap(a, b);
for(rl k=20; k >= 0; -- k)
if(dep[fa[a][k]] >= dep[b])
a = fa[a][k];
if(a == b) return a;
for(rl k=20; k >= 0; -- k)
if(fa[a][k] != fa[b][k])
a = fa[a][k], b = fa[b][k];
return fa[a][0];
}
tarjan求LCA
这是一种离线做法(将所有的询问存下来,然后再一一输出)。
这种做法我自我感觉不太好用,但是因为这种做法的时间复杂度是 \(O(n + m)\) 的,所以说有的人用起来很香(个人不喜欢,主要是没怎么敲过/
#include <bits/stdc++.h>
using namespace std;
#define ll int
#define rl register ll
const ll N = 20010, M = 2 * N;
ll n, m;
ll tot, ne[M], e[M], h[N], w[M];
ll p[N], dis[N], st[N];
ll res[N];
vector<pair<int, int>> query[N];
inline void add(ll a, ll b, ll c)
{
ne[++tot] = h[a], h[a] = tot, e[tot] = b, w[tot] = c;
}
inline void dfs(ll u, ll fa)
{
for(rl i=h[u]; ~i; i = ne[i])
{
ll v = e[i];
if(v == fa) continue;
dis[v] = dis[u] + w[i];
dfs(v, u);
}
}
inline ll find(ll x)
{
if(p[x] == x) return x;
else return p[x] = find(p[x]);
}
inline void tarjan(ll u)
{
st[u] = 1;
for(rl i=h[u]; ~i; i = ne[i])
{
ll v = e[i];
if(!st[v])
{
tarjan(v);
p[v] = u;
}
}
for(auto item : query[u])
{
ll y = item.first, id = item.second;
if(st[y] == 2)
{
ll anc = find(y);
res[id] = dis[y] + dis[u] - 2 * dis[anc];
}
}
st[u] = 2;
}
int main()
{
cin >> n >> m;
memset(h, -1, sizeof h);
for(rl i=1; i <= n - 1; ++ i)
{
ll a, b, c;
cin >> a >> b >> c;
add(a, b, c), add(b, a, c);
}
for(rl i=1; i <= m; ++ i)
{
ll a, b;
cin >> a >> b;
if(a != b)
{
query[a].push_back({b, i});
query[b].push_back({a, i});
}
}
for(rl i=1; i <= n; ++ i) p[i] = i;
dfs(1, -1);
tarjan(1);
for(rl i=1; i <= m; ++ i)
cout << res[i] << endl;
return 0;
}
O(1) 复杂度求LCA
注意,如果你会对这个图删删减减,加加变变,那么这个变化之后就不能直接查询了(st表带来的)。
我们首先求出来树上每个点的 \(dfs\) 序列。
我们考虑 \(u\) 和 \(v\) 之间有什么,令 \(lca(u, v) = x, dfn_u < dfn_v\)。那么 \(x\) 到 \(v\) 路径上的一点,一定是 \(u \to v\) 路径上一点,并且这个点是 \(u \to v\) 深度最小的,这个点的父亲节点就是 \(x\)。
再考虑一种特殊情况,当\(u\) 就是 \(v\) 的祖先的时候,深度最小得点就变成了\(u\), 为了规避这种情况,我们就选择在 \([dfn_u + 1 \to dfn_v]\) 上来查询。
区间深度最小可以使用 ST表 来维护。
下面是代码,也建议背过。
inline ll get(ll a, ll b) { return dep[a] < dep[b] ? a : b; }
inline void dfs(ll u, ll fa)
{
dfn[u] = ++ idx, st[0][idx] = fa, dep[u] = dep[fa] + 1;
for(rl i=h[u]; ~i; i = ne[i])
{
ll v = e[i];
if(v == fa) continue;
dfs(v, u);
}
}
inline void init()
{
dfs(1, -1);
for(rl i=1; (1 << i) <= n; ++ i)
for(rl j=1; j <= n - (1 << i) + 1; ++ j)
st[i][j] = get(st[i-1][j], st[i-1][j + (1 << i - 1)]);
}
inline ll lca(ll a, ll b)
{
if(a == b) return a;
a = dfn[a], b = dfn[b];
if(a > b) swap(a, b);
ll l = __lg(b - a); // 本人亲测用__lg(b - a),和log2(b - a) 都可以,这两个得区别可以上网搜。
return get(st[l][a + 1], st[l][b - (1 << l) + 1]);
}
浅谈关于LCA的更多相关文章
- 浅谈求lca
lca即最近公共祖先,求最近公共祖先的方法大概有3种,其实是窝只听说过3种,这3种做法分别是倍增求lca,树剖求lca和tarjan求lca,但是窝只会前2种,所以这里只说前2种算法了. 首先是倍增求 ...
- 浅谈倍增LCA
题目链接:https://www.luogu.org/problemnew/show/P3379 刚学了LCA,写篇blog加强理解. LCA(Least Common Ancestors),即最近公 ...
- 莫队浅谈&题目讲解
莫队浅谈&题目讲解 一.莫队的思想以及莫队的前置知识 莫队是一种离线的算法,他的实现借用了分块的思想.在学习莫队之前,本人建议学习一下分块,并对其有一定的理解. 二.莫队 现给出一道例题:bz ...
- 浅谈 Fragment 生命周期
版权声明:本文为博主原创文章,未经博主允许不得转载. 微博:厉圣杰 源码:AndroidDemo/Fragment 文中如有纰漏,欢迎大家留言指出. Fragment 是在 Android 3.0 中 ...
- 浅谈 LayoutInflater
浅谈 LayoutInflater 版权声明:本文为博主原创文章,未经博主允许不得转载. 微博:厉圣杰 源码:AndroidDemo/View 文中如有纰漏,欢迎大家留言指出. 在 Android 的 ...
- 浅谈Java的throw与throws
转载:http://blog.csdn.net/luoweifu/article/details/10721543 我进行了一些加工,不是本人原创但比原博主要更完善~ 浅谈Java异常 以前虽然知道一 ...
- 浅谈SQL注入风险 - 一个Login拿下Server
前两天,带着学生们学习了简单的ASP.NET MVC,通过ADO.NET方式连接数据库,实现增删改查. 可能有一部分学生提前预习过,在我写登录SQL的时候,他们鄙视我说:“老师你这SQL有注入,随便都 ...
- 浅谈WebService的版本兼容性设计
在现在大型的项目或者软件开发中,一般都会有很多种终端, PC端比如Winform.WebForm,移动端,比如各种Native客户端(iOS, Android, WP),Html5等,我们要满足以上所 ...
- 浅谈angular2+ionic2
浅谈angular2+ionic2 前言: 不要用angular的语法去写angular2,有人说二者就像Java和JavaScript的区别. 1. 项目所用:angular2+ionic2 ...
- iOS开发之浅谈MVVM的架构设计与团队协作
今天写这篇博客是想达到抛砖引玉的作用,想与大家交流一下思想,相互学习,博文中有不足之处还望大家批评指正.本篇博客的内容沿袭以往博客的风格,也是以干货为主,偶尔扯扯咸蛋(哈哈~不好好工作又开始发表博客啦 ...
随机推荐
- C++面试八股文:static_cast了解一下?
某日二师兄参加XXX科技公司的C++工程师开发岗位第20面: 面试官:C++中支持哪些类型转换? 二师兄:C++支持C风格的类型转换,并在C++11引入新的关键字规范了类型转换. 二师兄:C++11引 ...
- 解决Mysql 5.7 不能插入中文的问题
问题的解决方案 问题描述 : 在学习DML插入中文数据时 , 发现出现了以下问题 -- 插入数据 insert into tea (id , name) values (2 , '徐凤年'); -- ...
- 使用Flask和Django构建Web应用程序:现代Web应用程序框架
目录 1. 引言 2. 技术原理及概念 2.1 基本概念解释 2.2 技术原理介绍 2.3 相关技术比较 3. 实现步骤与流程 3.1 准备工作:环境配置与依赖安装 3.2 核心模块实现 3.3 集成 ...
- [ARM 汇编]高级部分—性能优化与调试—3.4.1 性能分析与优化策略
性能优化是嵌入式系统开发中的一个重要环节,尤其是在资源受限的环境下.性能优化的目标是提高代码执行速度.降低功耗和减少内存占用.在本章节中,我们将讨论性能分析与优化策略,并通过实例来学习如何应用这些策略 ...
- React后台管理系统(TypeScript、Redux状态管理)环境搭建01
搭建环境的时候,我们必须要先确保环境有node环境和npm环境,如下使用cmd命令 确保自己有了这两个环境之后我们就可以开始搭建项目,首先找一个文件夹,这个文件夹用来初始化当前环境,例如,我这里选 ...
- TCP/IP协议发明人G-Cerf
如果你是一个IT人,你可以不知道Vinton G. Cerf博士,但你不可能不知道TCP/IP; 如果你不是一个IT人,你可以不知道TCP/IP,但你不可能不知道互联网; 如果从1973年起,Vint ...
- mac shell终端命令行快捷键
Ctrl + d 删除一个字符,相当于通常的Delete键(命令行若无所有字符,则相当于exit:处理多行标准输入时也表示eof) Ctrl + h 退格删除一个字符,相当于通常的Backspace键 ...
- kubernetes(k8s):解决不在同一网段加入集群失败问题
执行下面命令,将内外网进行映射. iptables -t nat -A OUTPUT -d 10.140.128.121 -j DNAT --to-destination 10.170.129.153 ...
- 【Shell】函数
函数 1.语法 函数定义语法如下: [ function ] funname [()] { action; [return int;] } 说明 : 函数定义时,function 关键字可有可无. 函 ...
- CMOS信噪比与感光面积的关系
前言 一般情况下,相同分辨率的sensor,如果感光面积越大,则其单位像素的感光面积也越大,成像质量也会越好.即相同分辨率品质相当的sensor,2/3"的传感器成像质量一般情况就要优于1/ ...