【模板】Tarjian求LCA】的更多相关文章

概念 公共祖先,就是两个节点在这棵树上深度最大的公共的祖先节点 举个例子吧,如下图所示4和5的最近公共祖先是2,5和3的最近公共祖先是1,2和1的最近公共祖先是1. 算法 常用的求LCA的算法有:Tarjan/DFS+ST/倍增 其中 :ST和倍增都是在线的:Taijian是离线的 这里介绍离线的Tarjian算法 基本思想: 1.任选一个点为根节点,从根节点开始 2.遍历该点u所有子节点v,并标记这些子节点v已被访问过 3.若是v还有子节点,返回2,否则下一步 4.合并v到u上 5.寻找与当前…
看了好多dalao的博客,就总结一下啦ovo tarjian算法很是神奇,它的作用是求lca.它是一种离线算法. 在线是指输入一个询问输出一个结果. 离线是将询问一次性输入,一起处理. tarjan它是将m个询问打乱顺序,在每个结点上挂上它的询问,利用dfs和并查集进行处理. 对于一个结点u,如果要找(u,v)的lca,u和v的关系分为以下几种情况: (1)结点v在u的子树中,那么lca(u,v)=u; (2)  结点v不在u的子树中,那么v就在u子结点以外的结点中.v可能在u的爸爸的子树中,那…
//233 模板 LCA void dfs(int x,int f){ for(int i=0;i<E[x].size();i++){ int v = E[x][i]; if(v==f)continue; deep[v]=deep[x]+1; lca[v][0]=x; for(int j=1;j<maxm;j++){ int fa = lca[v][j-1]; if(fa==0)continue; lca[v][j]=lca[fa][j-1]; } dfs(v,x); } } int up(i…
倍增LCA \(fa[a][i]\)代表a的第\(2^{i}\)个祖先. 主体思路是枚举二进制位,让两个查询节点跳到同一高度然后再向上跳相同高度找LCA. int fa[N][21], dep[N]; void dfs(int u, int f) { dep[u] = dep[f] + 1, fa[u][0] = f; for (int i = 1;i <= 20;i++) fa[u][i] = fa[fa[u][i - 1]][i - 1]; for (int i = head[u];i;i…
倍增求lca模板 https://www.luogu.org/problem/show?pid=3379 #include<cstdio> #include<iostream> #include<cmath> #include<cstring> using namespace std; int t,n,cnt,m; int x,y; ][],p,root; ]; ]; ]; ; struct node { int next,to; }e[*]; inline…
定义LCA,最近公共祖先,是指一棵树上两个节点的深度最大的公共祖先.也可以理解为两个节点之间的路径上深度最小的点.我们这里用了倍增的方法求了LCA.我们的基本的思路就是,用dfs遍历求出所有点的深度.f[i][j]数组用来求的是距离节点i,距离2^j的祖先.可以知道,f[i][0]就是它的直接父亲.然后通过倍增的思路求出father数组的所有元素.然后进行lca.求lca的基本思路是:先让深度较大的点向上跳,然后x和y再同时向上跳2的幂,总会跳到这样两个点,他们的父亲结点是同一个点,那就是x和y…
倍增求 LCA 是在线的,而且比 ST 好写多了,理解起来比 ST 和 Tarjan 都容易,于是就自行脑补吧,代码写得容易看懂 关键理解 f[i][j] 表示 i 号节点的第 2j 个父亲,也就是往上走 2j 个节点 求 LCA 的时候先倍增让两点深度一样,再倍增求 另外丢两个链接,这两个有详细讲解 ST 算法 http://www.cnblogs.com/hadilo/p/5837517.html Tarajan 算法 http://www.cnblogs.com/hadilo/p/5840…
原理可以参考大神 LCA_Tarjan (离线) TarjanTarjan 算法求 LCA 的时间复杂度为 O(n+q) ,是一种离线算法,要用到并查集.(注:这里的复杂度其实应该不是 O(n+q) ,还需要考虑并查集操作的复杂度 ,但是由于在多数情况下,路径压缩并查集的单次操作复杂度可以看做 O(1),所以写成了 O(n+q).) #include<bits/stdc++.h> using namespace std; ; ]; struct Query{ int v, id; Query(…
倍增求\(LCA\) 倍增基础 从字面意思理解,倍增就是"成倍增长". 一般地,此处的增长并非线性地翻倍,而是在预处理时处理长度为\(2^n(n\in \mathbb{N}^+)\)的区间值.在这些预处理结果的基础上,我们可以进一步求出任意长度区间的答案. 比如区间最值问题\((RMQ)\)就可以使用倍增解决.对于每个起始点,预处理长度为\(2^n\)的区间最值.之后每段区间都可以以此求出,如: \(f(1,7)=\max(f(1,4),f(3,7))\) 以上是最简单的一个举例.在计…
树链剖分中各种数组的作用: siz[]数组,用来保存以x为根的子树节点个数 top[]数组,用来保存当前节点的所在链的顶端节点 son[]数组,用来保存重儿子 dep[]数组,用来保存当前节点的深度 fa[]数组,用来保存当前节点的父亲 tid[]数组,用来保存树中每个节点剖分后的新编号 rank[]数组,用来保存当前节点在线段树中的位置 树链剖分求LCA据说很快QWQ,反正我在洛谷上评测的时候比倍增整整快了3分之1. 蓝后我们来说怎么用树链剖分求: 1,第一种情况我们要比较的数在一条链上,比如…