二叉树中两个指定节点的最近公共祖先

二叉树中两个指定节点的最近公共祖先

236. 二叉树的最近公共祖先 - 力扣(LeetCode)

在二叉树中,两个节点的最近公共祖先(LCA)是指同时是这两个节点的祖先的最低节点(即深度最大的共同祖先)。例如,给定一个二叉树,节点A和节点B的LCA是树中第一个同时包含A和B在其子树中的节点。这个问题可以通过递归遍历树来解决,核心思想是利用后序遍历来检查节点之间的关系。

解题思路
  • 递归方法:从根节点开始,递归地遍历左子树和右子树。如果当前节点是其中一个目标节点(p或q),则返回该节点;如果在左子树和右子树中都找到了目标节点,则当前节点就是LCA;否则,返回非null的子节点结果。
  • 关键点:算法基于后序遍历(左-右-根),因为它需要先处理子树,再处理根节点,从而自底向上地确定LCA。
  • 边界条件:如果树为空或目标节点不存在,返回null;如果目标节点是根节点,则根节点就是LCA。
算法流程
  1. 基本检查
    • 如果当前节点为null,返回null。
    • 如果当前节点是p或q中的一个,返回当前节点(表示找到了一个目标节点)。
  2. 递归遍历
    • 递归地在左子树中查找p和q。
    • 递归地在右子树中查找p和q。
  3. 结果合并
    • 如果左子树和右子树的递归结果都非null,表示当前节点是LCA(因为p和q分布在左右子树中)。
    • 如果只有左子树结果非null,返回左子树结果(表示LCA在左子树中)。
    • 如果只有右子树结果非null,返回右子树结果(表示LCA在右子树中)。
    • 如果都null,返回null(表示未找到)。

该算法的时间复杂度为O(n),其中n是树中的节点数,因为每个节点被访问一次。空间复杂度为O(h),其中h是树的高度,主要用于递归栈。

Java实现

以下是Java代码实现。假设树节点定义为TreeNode类,包含值、左子节点和右子节点。

java 复制代码
class TreeNode {
    int val;
    TreeNode left;
    TreeNode right;
    TreeNode(int x) { val = x; }
}

public class Solution {
    public TreeNode lowestCommonAncestor(TreeNode root, TreeNode p, TreeNode q) {
        // 边界条件:如果根节点为空,或根节点就是p或q,直接返回根节点
        if (root == null || root == p || root == q) {
            return root;
        }
        
        // 递归遍历左子树和右子树
        TreeNode left = lowestCommonAncestor(root.left, p, q);
        TreeNode right = lowestCommonAncestor(root.right, p, q);
        
        // 如果左右子树都非null,当前节点是LCA
        if (left != null && right != null) {
            return root;
        }
        // 否则,返回非null的子节点结果
        return left != null ? left : right;
    }
}
C语言实现

在C语言中,我们定义树节点结构体,并实现递归函数。注意内存管理。

c 复制代码
#include <stdlib.h>

struct TreeNode {
    int val;
    struct TreeNode *left;
    struct TreeNode *right;
};

struct TreeNode* lowestCommonAncestor(struct TreeNode* root, struct TreeNode* p, struct TreeNode* q) {
    // 边界条件:如果根节点为空,或根节点就是p或q,直接返回根节点
    if (root == NULL || root == p || root == q) {
        return root;
    }
    
    // 递归遍历左子树和右子树
    struct TreeNode* left = lowestCommonAncestor(root->left, p, q);
    struct TreeNode* right = lowestCommonAncestor(root->right, p, q);
    
    // 如果左右子树都非null,当前节点是LCA
    if (left != NULL && right != NULL) {
        return root;
    }
    // 否则,返回非null的子节点结果
    return left != NULL ? left : right;
}
代码解释
  • 递归过程:函数从根节点开始,先检查当前节点是否为目标节点。如果不是,则递归地在左子树和右子树中查找。
  • 结果处理:递归返回后,如果左右子树都有结果(即非null),则当前节点是LCA;否则,返回有结果的子树(表示LCA在该子树中)。
  • 正确性:算法保证了当p和q分布在当前节点的左右子树时,当前节点是LCA;如果p和q在同一个子树中,则递归返回该子树的结果。
  • 测试建议:在实际使用中,确保p和q在树中存在;可以通过构建二叉树并调用函数验证。
相关推荐
ppo922 小时前
Spring Boot 集成 Kafka 3.9.0:部署、监控与消息发送教程
java·架构
JavaEdge.2 小时前
IDEA卡死没反应的全部解决方案
java·ide·intellij-idea
高山上有一只小老虎2 小时前
使用Memory Analyzer (MAT)分析内存溢出
java·jvm
嘴贱欠吻!2 小时前
JavaSE基础知识
java·开发语言
逝水如流年轻往返染尘3 小时前
Java输入输出
java·开发语言
Alsn863 小时前
29.Java中常见加解密算法的基本实现
java·开发语言·算法
沉默王二3 小时前
TRAE+Gemini,成为我解读 Agent 微服项目的最佳工具
java·后端·程序员
多则惑少则明3 小时前
SpringBoot3整合knife4j(swagger3)
java·spring boot·swagger
星月昭铭3 小时前
Spring Boot写一个/v1/chat/completions接口给Cherry Studio流式调用
java·spring boot·后端·ai