首页 > 其他分享 >树上倍增法求LCA

树上倍增法求LCA

时间:2023-02-19 21:32:08浏览次数:26  
标签:结点 int tu fa deg LCA 倍增 root 法求

我们找的是任意两个结点的最近公共祖先, 那么我们可以考虑这么两种种情况:


1.两结点的深度相同.

2.两结点深度不同.

第一步都要转化为情况1,这种可处理的情况。


先不考虑其他, 我们思考这么一个问题: 对于两个深度不同的结点, 把深度更深的那个向其父节点迭代, 直到这个迭代结点和另一个结点深度相同, 那么这两个深度相同的结点的Lca也就是原两个结点的Lca. 因此第二种情况转化成第一种情况来求解Lca是可行的. 这里我们使用倍增法以最快的速度找到相同的深度,然后开始求LCA。求LCA使用倍增法,倍增的条件是找到相同的祖先,减小步距

 

/*
* LCA在线算法(倍增法)
*/
const int MAXN = 10010;
const int DEG = 20;

struct Edge
{
int to, next;
} edge[MAXN * 2];

int head[MAXN], tot;
void addedge(int u, int v)
{
edge[tot].to = v;
edge[tot].next = head[u];
head[u] = tot++;
}

void init()
{
tot = 0;
memset(head, -1, sizeof(head));
}

int fa[MAXN][DEG]; // fa[i][j]表示结点i的第2^j个祖先
int deg[MAXN]; // 深度数组

void BFS(int root)
{
queue<int>que;
deg[root] = 0;
fa[root][0] = root;
que.push(root);
while (!que.empty())
{
int tmp = que.front();
que.pop();
for (int i = 1; i < DEG; i++)
{
fa[tmp][i] = fa[fa[tmp][i - 1]][i - 1];
}
for (int i = head[tmp]; i != -1; i = edge[i].next)
{
int v = edge[i].to;
if (v == fa[tmp][0])
{
continue;
}
deg[v] = deg[tmp] + 1;
fa[v][0] = tmp;
que.push(v);
}
}
}

int LCA(int u, int v)
{
if (deg[u] > deg[v])
{
swap(u, v);
}
int hu = deg[u], hv = deg[v];
int tu = u, tv = v;
for (int det = hv-hu, i = 0; det ; det >>= 1, i++)
{
if (det & 1)
{
tv = fa[tv][i];
}
}
if (tu == tv)
{
return tu;
}
for (int i = DEG - 1; i >= 0; i--)
{
if (fa[tu][i] == fa[tv][i])
{
continue;
}
tu = fa[tu][i];
tv = fa[tv][i];
}
return fa[tu][0];
}

bool flag[MAXN];

int main()
{
int T;
int n;
int u, v;
scanf("%d", &T);

while(T--)
{
scanf("%d", &n);
init();
memset(flag, false, sizeof(flag));
for (int i = 1; i < n; i++)
{
scanf("%d%d", &u, &v);
addedge(u, v);
addedge(v, u);
flag[v] = true;
}
int root;
for (int i = 1; i <= n; i++)
{
if (!flag[i])
{
root = i;
break;
}
}
BFS(root);
scanf("%d%d", &u, &v);
printf("%d\n", LCA(u, v));
}
return 0;
}

 

标签:结点,int,tu,fa,deg,LCA,倍增,root,法求
From: https://blog.51cto.com/u_14682436/6066837

相关文章

  • 【路径规划】基于遗传算法求解单向交通流分配优化问题附matlab代码
    ✅作者简介:热爱科研的Matlab仿真开发者,修心和技术同步精进,matlab项目合作可私信。......
  • 【c++/c】幂法求解绝对值最大的特征值、最大和最小的特征值
    【part1:幂法的迭代格式】  【part2:计算绝对值最大特征值的步骤】1.明确要求解特征值的矩阵A和初始非零向量u02.将u0单位化得到yk-13.通过矩......
  • LCA 板子
    本例求树上两点的距离 #include<bits/stdc++.h>usingnamespacestd;constintN=5*1e6+2,M=5*N;intnxt[M],hd[N],all,go[M],n;intdep[N],f[N][22];void......
  • 快速傅里叶变换的倍增实现
    本文作者为JustinRochester。目录地址上一篇下一篇快速傅里叶变换的倍增实现......
  • C语言填空:减损法求最大公约数
    #include<stdio.h>//<<九章算术>>更相减损法:可以用来求两个数的最大公约数,即“可半者半之,不可半者,副置分母、子之数,以少减多,更相减损,求其等也。//以等数约之。///第......
  • 【代码源 Div1 - 105】#451. Dis(倍增求LCA)
    problemsolution给出n个点的一棵树,每个点有各自的点权,m次询问两个点简单路径所构成点集的异或和。直接在树上求LCA,把每个点权放进去预处理一下即可。#include<bits/stdc+......
  • 【bzoj4668】冷战 (并查集按秩合并+朴素LCA)
    题目描述1946年3月5日,英国前首相温斯顿·丘吉尔在美国富尔顿发表“铁幕演说”,正式拉开了冷战序幕。美国和苏联同为世界上的“超级大国”,为了争夺世界霸权,两国及其盟国......
  • 遗传算法求TSP问题
    一、实验内容及目的本实验以遗传算法为研究对象,分析了遗传算法的选择、交叉、变异过程,采用遗传算法设计并实现了商旅问题求解,解决了商旅问题求解最合适的路径,达到用遗传算......
  • 递归法求解数列组合的各种情况
    C#代码:staticvoidMain(string[]args){int[]items=newint[]{0,1,2,3,4};intm=3;List<int[]>allC......
  • Genius ACM(归并+倍增)
    题目描述给定一个整数 MM,对于任意一个整数集合 SS,定义“校验值”如下:从集合 SS 中取出 MM 对数(即 2∗M2∗M 个数,不能重复使用集合中的数,如果 SS 中的整数不够 ......