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的更多相关文章

  1. 浅谈求lca

    lca即最近公共祖先,求最近公共祖先的方法大概有3种,其实是窝只听说过3种,这3种做法分别是倍增求lca,树剖求lca和tarjan求lca,但是窝只会前2种,所以这里只说前2种算法了. 首先是倍增求 ...

  2. 浅谈倍增LCA

    题目链接:https://www.luogu.org/problemnew/show/P3379 刚学了LCA,写篇blog加强理解. LCA(Least Common Ancestors),即最近公 ...

  3. 莫队浅谈&题目讲解

    莫队浅谈&题目讲解 一.莫队的思想以及莫队的前置知识 莫队是一种离线的算法,他的实现借用了分块的思想.在学习莫队之前,本人建议学习一下分块,并对其有一定的理解. 二.莫队 现给出一道例题:bz ...

  4. 浅谈 Fragment 生命周期

    版权声明:本文为博主原创文章,未经博主允许不得转载. 微博:厉圣杰 源码:AndroidDemo/Fragment 文中如有纰漏,欢迎大家留言指出. Fragment 是在 Android 3.0 中 ...

  5. 浅谈 LayoutInflater

    浅谈 LayoutInflater 版权声明:本文为博主原创文章,未经博主允许不得转载. 微博:厉圣杰 源码:AndroidDemo/View 文中如有纰漏,欢迎大家留言指出. 在 Android 的 ...

  6. 浅谈Java的throw与throws

    转载:http://blog.csdn.net/luoweifu/article/details/10721543 我进行了一些加工,不是本人原创但比原博主要更完善~ 浅谈Java异常 以前虽然知道一 ...

  7. 浅谈SQL注入风险 - 一个Login拿下Server

    前两天,带着学生们学习了简单的ASP.NET MVC,通过ADO.NET方式连接数据库,实现增删改查. 可能有一部分学生提前预习过,在我写登录SQL的时候,他们鄙视我说:“老师你这SQL有注入,随便都 ...

  8. 浅谈WebService的版本兼容性设计

    在现在大型的项目或者软件开发中,一般都会有很多种终端, PC端比如Winform.WebForm,移动端,比如各种Native客户端(iOS, Android, WP),Html5等,我们要满足以上所 ...

  9. 浅谈angular2+ionic2

    浅谈angular2+ionic2   前言: 不要用angular的语法去写angular2,有人说二者就像Java和JavaScript的区别.   1. 项目所用:angular2+ionic2 ...

  10. iOS开发之浅谈MVVM的架构设计与团队协作

    今天写这篇博客是想达到抛砖引玉的作用,想与大家交流一下思想,相互学习,博文中有不足之处还望大家批评指正.本篇博客的内容沿袭以往博客的风格,也是以干货为主,偶尔扯扯咸蛋(哈哈~不好好工作又开始发表博客啦 ...

随机推荐

  1. CF1817E Half-sum

    题意 有一个大小为 \(N\) 的非负整数集合 \(A\),每次你可以从集合中取任意两个数,并将它们的平均数放回序列.不停操作,知道集合最后剩下两个数.请求出这两个数的差的绝对值的最大值对 \(10^ ...

  2. 基于VAE的风险分析:基于历史数据的风险分析、基于实时数据的风险分析

    目录 引言 随着人工智能和机器学习的发展,风险分析已经成为许多行业和组织中不可或缺的一部分.传统的基于经验和规则的风险分析方法已经难以满足现代风险分析的需求,因此基于VAE的风险分析方法逐渐成为了主流 ...

  3. React框架学习基础篇-HelloReact-01

    一直想掌握一门前端技术,于是想跟着张天宇老师学习,便开始学习React,以此来记录一下我的学习之旅. 学习一门新的技术首先是去官网看看,React官网链接是[https://zh-hans.react ...

  4. gitlab配置环境及pycharm配置

    一.gitlab介绍 GitLab 是一个用于仓库管理系统的开源项目,使用Git作为代码管理工具,并在此基础上搭建起来的web服务 git.gitlab.GitHub的简单区别 git 是一种基于命令 ...

  5. 关于 axios 是什么?以及怎么用?

    〇.前言 Axios 是一个基于 promise 的 HTTP 库,可以用在浏览器和 Node.js 中.简单的讲就是可以发送 Get.Post 请求. 诸如 Vue.React.Angular 等前 ...

  6. 2022蓝桥杯B组(java)版

    2022蓝桥杯b组 A题 import java.math.BigInteger; public class A { public static void main(String[] args) { ...

  7. 【小小Demo】微信公众号如何接入微信机器人

    微信对话开放平台文档 官方文档 平台简介 微信对话开放平台开放了微信在对话领域积累多年的的智能对话技术,开发者及非开发者可简单.快速地搭建智能对话机器人(智能客服), 并接入公众号.小程序等,为业务赋 ...

  8. Day01_Java作业

    A:选择题 1:下列标识符哪个是合法的(a) A.class B.$abc C.1234 D.Car.taxi B:填空题 1: java源程序的扩展名是( .java ) 2: java程序经编译后 ...

  9. Ubuntu16.04配置NTP时间同步

    环境 查看系统版本:lsb_release -a 名词解释 PDT是指太平洋夏令时(Pacific Daylight Time),是美国西部地区和加拿大的一部分地区使用的时区.它位于UTC-7和UTC ...

  10. 基于GPT搭建私有知识库聊天机器人(六)仿chatGPT打字机效果

    文章链接: 基于GPT搭建私有知识库聊天机器人(一)实现原理 基于GPT搭建私有知识库聊天机器人(二)环境安装 基于GPT搭建私有知识库聊天机器人(三)向量数据训练 基于GPT搭建私有知识库聊天机器人 ...