面试题 04.08. 首个共同祖先
大约 2 分钟
面试题 04.08. 首个共同祖先
- 标签:树、深度优先搜索、二叉树
- 难度:中等
题目链接
题目大意
给定一个二叉树,要求找到该树中指定节点 p
、q
的最近公共祖先:
祖先:若节点
p
在节点node
的左子树或右子树中,或者p = node
,则称node
是p
的祖先。最近公共祖先:对于树的两个节点
p
、q
,最近公共祖先表示为一个节点lca_node
,满足lca_node
是p
、q
的祖先且lca_node
的深度尽可能大(一个节点也可以是自己的祖先)。
解题思路
设 lca_node
为节点 p
、q
的最近公共祖先。则 lca_node
只能是下面几种情况:
p
、q
在lca_node
的子树中,且分别在lca_node
的两侧子树中。p == lca_node
,且q
在lca_node
的左子树或右子树中。q == lca_node
,且p
在lca_node
的左子树或右子树中。
下面递归求解 lca_node
。递归需要满足以下条件:
- 如果
p
、q
都不为空,则返回p
、q
的公共祖先。 - 如果
p
、q
只有一个存在,则返回存在的一个。 - 如果
p
、q
都不存在,则返回存在的一个。
具体思路为:
- 如果当前节点
node
为None
,则说明p
、q
不在node
的子树中,不可能为公共祖先,直接返回None
。 - 如果当前节点
node
等于p
或者q
,那么node
就是p
、q
的最近公共祖先,直接返回node
。 - 递归遍历左子树、右子树,并判断左右子树结果。
- 如果左子树为空,则返回右子树。
- 如果右子树为空,则返回左子树。
- 如果左右子树都不为空,则说明
p
、q
在当前根节点的两侧,当前根节点就是他们的最近公共祖先。 - 如果左右子树都为空,则返回空。
代码
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