最近公共祖先详解

一、定位概念

最近公共祖先(LCA),指的是在一个树或有向无环图中,找到两个节点的共同祖先中深度最小的一个。

LCA问题是计算机科学中一种常见的算法问题,它有着广泛的应用。例如,可以用来处理家谱信息,或者解决大多数其他相关问题,例如计算两个版本之间的最近公共祖先或两个字符串之间的LCS(最长公共子序列)等等。

二、基本算法

LCA问题的本质是求树中两个节点的最近公共祖先,最简单的算法是遍历整棵树,记录下每个节点的祖先,比较两个节点的祖先列表,找到最后一个相同的节点即为最近公共祖先。该算法的时间复杂度为O(N^2),其中N为节点的数量。

#include <bits/stdc++.h>
using namespace std;
int n, m, u, v, root; //输入数据:n表示树的节点数量,m表示询问数量,u、v表示两个节点
vector<int> adj[10001]; //存储树的连接关系
vector<int> parents[10001]; //存储每个节点的祖先
void dfs(int u, int p) { //深度优先遍历
    parents[u].push_back(p);
    for (auto v : adj[u]) {
        if (v != p) dfs(v, u);
    }
}
int lca(int u, int v) { //计算最近公共祖先
    if (parents[u].size() > parents[v].size()) swap(u, v);
    int len1 = parents[u].size();
    int len2 = parents[v].size();
    for (int i = 0; i = 0; i--) {
        if (parents[u][i] == parents[v][i]) return parents[u][i];
    }
    return root;
}
int main() {
    scanf("%d%d%d", &n, &m, &root);
    for (int i = 1; i < n; i++) {
        scanf("%d%d", &u, &v);
        adj[u].push_back(v);
        adj[v].push_back(u);
    }
    dfs(root, 0); //预处理每个节点的祖先
    for (int i = 1; i <= m; i++) {
        scanf("%d%d", &u, &v);
        printf("%d\n", lca(u, v));
    }
    return 0;
}

三、改善算法

遍历整棵树是一种时间复杂度比较高的算法,可以对其进行改善,以达到更好的效果。改善算法的核心思想是预处理出每个节点到根节点的距离,然后通过二分查找的方法,找到两个节点深度最浅的公共祖先。

#include <bits/stdc++.h>
using namespace std;
const int N = 10001;
int n, m, u, v, root, depth[N], f[N][16]; //f数组用于辅助计算节点的距离
vector<int> adj[N];
void dfs(int u, int p) { //深度优先遍历,预处理每个节点的深度和父节点
    depth[u] = depth[p] + 1;
    f[u][0] = p;
    for (int i = 1; (1 << i) <= depth[u]; i++) {
        f[u][i] = f[f[u][i - 1]][i - 1]; //使用DP的思想,把节点的距离预处理出来
    }
    for (auto v : adj[u]) {
        if (v != p) dfs(v, u);
    }
}
int lca(int u, int v) { //利用二分查找的思想,计算最近公共祖先
    if (depth[u]  0; i++) {
        if (diff & 1) u = f[u][i];
        diff >>= 1;
    }
    if (u == v) return u;
    for (int i = log2(depth[u]); i >= 0; i--) {
        if (f[u][i] != f[v][i]) {
            u = f[u][i];
            v = f[v][i];
        }
    }
    return f[u][0];
}
int main() {
    scanf("%d%d%d", &n, &m, &root);
    for (int i = 1; i < n; i++) {
        scanf("%d%d", &u, &v);
        adj[u].push_back(v);
        adj[v].push_back(u);
    }
    dfs(root, 0);
    for (int i = 1; i <= m; i++) {
        scanf("%d%d", &u, &v);
        printf("%d\n", lca(u, v));
    }
    return 0;
}

四、总结

本文从概念、基本算法和改善算法三个方面进行了LCA问题的详细阐述。LCA问题是计算机科学中一种常见的算法问题,在实际生活中有广泛的应用。

基础算法的核心思想是遍历整棵树,记录节点的祖先列表,计算节点的深度,比较两个节点的祖先列表,找到最后一个相同的节点即为最近公共祖先。

改善算法的核心思想是预处理出每个节点到根节点的距离,使用二分查找的方法,找到两个节点深度最浅的公共祖先,达到了比基础算法更高效的效果。

原创文章,作者:OUWUA,如若转载,请注明出处:https://www.506064.com/n/332176.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
OUWUAOUWUA
上一篇 2025-01-21 17:30
下一篇 2025-01-21 17:30

相关推荐

  • Linux sync详解

    一、sync概述 sync是Linux中一个非常重要的命令,它可以将文件系统缓存中的内容,强制写入磁盘中。在执行sync之前,所有的文件系统更新将不会立即写入磁盘,而是先缓存在内存…

    编程 2025-04-25
  • 神经网络代码详解

    神经网络作为一种人工智能技术,被广泛应用于语音识别、图像识别、自然语言处理等领域。而神经网络的模型编写,离不开代码。本文将从多个方面详细阐述神经网络模型编写的代码技术。 一、神经网…

    编程 2025-04-25
  • Linux修改文件名命令详解

    在Linux系统中,修改文件名是一个很常见的操作。Linux提供了多种方式来修改文件名,这篇文章将介绍Linux修改文件名的详细操作。 一、mv命令 mv命令是Linux下的常用命…

    编程 2025-04-25
  • Python输入输出详解

    一、文件读写 Python中文件的读写操作是必不可少的基本技能之一。读写文件分别使用open()函数中的’r’和’w’参数,读取文件…

    编程 2025-04-25
  • nginx与apache应用开发详解

    一、概述 nginx和apache都是常见的web服务器。nginx是一个高性能的反向代理web服务器,将负载均衡和缓存集成在了一起,可以动静分离。apache是一个可扩展的web…

    编程 2025-04-25
  • MPU6050工作原理详解

    一、什么是MPU6050 MPU6050是一种六轴惯性传感器,能够同时测量加速度和角速度。它由三个传感器组成:一个三轴加速度计和一个三轴陀螺仪。这个组合提供了非常精细的姿态解算,其…

    编程 2025-04-25
  • 详解eclipse设置

    一、安装与基础设置 1、下载eclipse并进行安装。 2、打开eclipse,选择对应的工作空间路径。 File -> Switch Workspace -> [选择…

    编程 2025-04-25
  • Python安装OS库详解

    一、OS简介 OS库是Python标准库的一部分,它提供了跨平台的操作系统功能,使得Python可以进行文件操作、进程管理、环境变量读取等系统级操作。 OS库中包含了大量的文件和目…

    编程 2025-04-25
  • Java BigDecimal 精度详解

    一、基础概念 Java BigDecimal 是一个用于高精度计算的类。普通的 double 或 float 类型只能精确表示有限的数字,而对于需要高精度计算的场景,BigDeci…

    编程 2025-04-25
  • git config user.name的详解

    一、为什么要使用git config user.name? git是一个非常流行的分布式版本控制系统,很多程序员都会用到它。在使用git commit提交代码时,需要记录commi…

    编程 2025-04-25

发表回复

登录后才能评论