跳至主要內容

剑指 Offer 68 - II. 二叉树的最近公共祖先

ITCharge大约 2 分钟

剑指 Offer 68 - II. 二叉树的最近公共祖先open in new window

  • 标签:树、深度优先搜索、二叉树
  • 难度:简单

题目链接

题目大意

给定一个二叉树的根节点 root,再给定两个指定节点 pq

要求:找到两个指定节点 pq 的最近公共祖先。

  • 祖先:若节点 p 在节点 node 的左子树或右子树中,或者 p == node,则称 nodep 的祖先。
  • 最近公共祖先:对于树的两个节点 pq,最近公共祖先表示为一个节点 lca_node,满足 lca_nodepq 的祖先且 lca_node 的深度尽可能大(一个节点也可以是自己的祖先)

解题思路

lca_node 为节点 pq 的最近公共祖先。则 lca_node 只能是下面几种情况:

  • pqlca_node 的子树中,且分别在 lca_node 的两侧子树中。
  • p = lca_node,且 qlca_node 的左子树或右子树中。
  • q = lca_node,且 plca_node 的左子树或右子树中。

下面递归求解 lca_node。递归需要满足以下条件:

  • 如果 pq 都不为空,则返回 pq 的公共祖先。
  • 如果 pq 只有一个存在,则返回存在的一个。
  • 如果 pq 都不存在,则返回存在的一个。

具体思路为:

  • 如果当前节点 nodeNone,则说明 pq 不在 node 的子树中,不可能为公共祖先,直接返回 None
  • 如果当前节点 node 等于 p 或者 q,那么 node 就是 pq 的最近公共祖先,直接返回 node
  • 递归遍历左子树、右子树,并判断左右子树结果。
    • 如果左子树为空,则返回右子树。
    • 如果右子树为空,则返回左子树。
    • 如果左右子树都不为空,则说明 pq 在当前根节点的两侧,当前根节点就是他们的最近公共祖先。
    • 如果左右子树都为空,则返回空。

代码

class Solution:
    def lowestCommonAncestor(self, root: TreeNode, p: TreeNode, q: TreeNode) -> TreeNode:
        if root == p or root == q:
            return root

        if root:
            node_left = self.lowestCommonAncestor(root.left, p, q)
            node_right = self.lowestCommonAncestor(root.right, p, q)
            if node_left and node_right:
                return root
            elif not node_left:
                return node_right
            else:
                return node_left
        return None