LeetCode 236. 二叉树的最近公共祖先(LCA)

LeetCode 236. 二叉树的最近公共祖先(LCA)

📌 题目链接


💡 解题思路

本题要求在一棵 普通二叉树 中找到两个节点 pq最近公共祖先(LCA)

最近公共祖先定义

对于有根树 T 的两个节点 p、q,最近公共祖先表示为一个节点 x,满足:

  • x 是 p 和 q 的祖先
  • x 的深度尽可能大(离 p、q 最近)

🌳 核心思想:后序遍历(自底向上)

这是一道非常经典的 后序遍历(Left → Right → Root) 题目。由于是普通二叉树,无法利用大小关系剪枝,因此采用 后序遍历(左右根)。 只有当左右子树的递归结果都已知时,才能判断当前节点是否为最近公共祖先。

递归逻辑

对于当前节点 root

  1. 递归终止条件

    • 如果 root == null,返回 null
    • 如果 root == proot == q,直接返回 root(不再向下递归)
  2. 递归左右子树

    java 复制代码
    TreeNode l = lowestCommonAncestor(root.left, p, q);
    TreeNode r = lowestCommonAncestor(root.right, p, q);
  3. 根据左右返回值判断结果

    • ✅ 左右都不为空 → root 就是 LCA
    • ✅ 左不为空,右为空 → 返回左
    • ✅ 右不为空,左为空 → 返回右
    • ❌ 都为空 → 返回 null

🧠 为什么这样是对的?

情况一:p 和 q 分别在左右子树

复制代码
        root
       /    \
      p      q
  • l != null, r != null
  • 只有 root 能同时看到它们
  • ✅ 返回 root

情况二:p 是 q 的祖先(或反过来)

复制代码
        p
         \
          q
  • 递归到 p 时直接返回
  • 上层不会再找到第二个节点
  • ✅ 返回 p

情况三:只在一个子树中

  • 另一个子树返回 null
  • 直接返回非空的那个结果

🧩 完整代码(Java)

java 复制代码
/**
 * Definition for a binary tree node.
 */
public class TreeNode {
    int val;
    TreeNode left;
    TreeNode right;
    TreeNode(int x) { val = x; }
}

class Solution {
    public TreeNode lowestCommonAncestor(TreeNode root, TreeNode p, TreeNode q) {
        // 1. 递归终止条件
        if (root == null) return null;
        if (root == p || root == q) return root;

        // 2. 递归左右子树
        TreeNode l = lowestCommonAncestor(root.left, p, q);
        TreeNode r = lowestCommonAncestor(root.right, p, q);

        // 3. 根据左右结果判断
        if (l != null && r != null) return root; // p、q 分布在两侧
        if (l == null && r != null) return r;
        if (l != null && r == null) return l;
        return null;
    }
}

⏱ 复杂度分析

指标 复杂度 说明
时间复杂度 O(n) 每个节点最多访问一次
空间复杂度 O(h) 递归栈深度,h 为树高
  • 最坏情况(链状树):O(n)
  • 平衡二叉树:O(log n)

🔍 常见误区

误区 正确理解
必须先找 p 再找 q ❌ 顺序无关
必须记录父节点 ❌ 递归天然回溯
只适用于 BST ❌ 本题是普通二叉树
返回第一个遇到的节点 ❌ 要看左右子树结果

✅ 总结一句话

后序遍历 + 左右返回值判断,是二叉树 LCA 问题的通用解法。


相关推荐
兰令水1 小时前
leecodecode【二叉树排序+最近公共祖先】【2026.6.2打卡-java版本】
java·数据结构·算法·leetcode
人道领域1 小时前
【LeetCode刷题日记】77&&216.回溯算法剪枝优化在组合问题中的应用
java·算法·leetcode
Deepoch1 小时前
Deepoc数学大模型:以低幻觉特性护航半导体精准设计与制造
大数据·人工智能·算法·半导体·deepoc
诸葛务农1 小时前
共沸脱水技术及其在光刻胶用PGMEA纯化中的应用(上)
java·数据库·算法
£suPerpanda1 小时前
AtCoder Beginner Contest 453
c++·算法
圣保罗的大教堂1 小时前
leetcode 3633. 最早完成陆地和水上游乐设施的时间 I 简单
leetcode
词元Max1 小时前
4.2 决策树与随机森林
算法·决策树·随机森林
菜菜的顾清寒1 小时前
力扣HOT100(51) 动态规划-单词拆分
算法·leetcode·动态规划
风筝在晴天搁浅1 小时前
剑指Offer LCR 143.子结构判断
算法