【二叉树算法题记录】236. 二叉树的最近公共祖先

题目链接

题目描述

给定一个二叉树, 找到该树中两个指定节点的最近公共祖先。

百度百科中最近公共祖先的定义为:“对于有根树 T 的两个节点 p、q,最近公共祖先表示为一个节点 x,满足 x 是 p、q 的祖先且 x 的深度尽可能大(一个节点也可以是它自己的祖先)。”

题目分析

这题显然是要从下往上去看,而我们知道二叉树的遍历是从根节点开始的,但是处理顺序可以从下往上。
在这里插入图片描述

  • 第一种情况:例如p=3, q=5p=0, q=3pq分别存在于最近公共祖先节点左子树右子树中。
  • 第二种情况:例如p=2, q=5或是p=4, q=5最近公共祖先节点就是pq

但无论是哪种情况,我们都要从下往上去找最近公共祖先节点。

第一种情况
对于第一种情况来说,我们首先要找到pq,然后向上传递,直到某节点的左右子树都包含pq,则这个节点就是他们的最近公共祖先。

  • 确定递归参数及返回值:因为我们想将p和q向上传递,所以返回类型仍然是TreeNode*。传入参数就是当前节点和p和q:
TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q)
  • 确定递归终止条件:如果当前节点为空,则返回空;如果当前节点是p或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);,然后总结每种情况并返回相应的值。这里包含以下四种情况:
    (1)如果该节点左右子树分别含有p或q,则当前节点是最近公共祖先,返回当前节点;
    if(left!=NULL && right!=NULL) return root;
    (2)如果当前节点左子树含有p或q,右子树不含,则返回左子树中含有的节点p或q;
    else if(left!=NULL && right==NULL) return left;
    (3)如果当前节点右子树含有p或q,左子树不含,则返回右子树中含有的节点p或q;
    else if(left==NULL && right!=NULL) return right;
    (4)如果左右子树都不包含,则返回空
    else return NULL;

整体cpp代码如下

/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     TreeNode *left;
 *     TreeNode *right;
 *     TreeNode(int x) : val(x), left(NULL), right(NULL) {}
 * };
 */

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);
        // 如果该节点左右子树分别含有p或q,则是最近公共祖先,返回当前节点;
        if(left!=NULL && right!=NULL) return root;
        // 如果当前节点左子树含有p或q,右子树不含,则返回左子树中含有的节点p或q;
        else if(left!=NULL && right==NULL) return left;
        // 如果当前节点右子树含有p或q,左子树不含,则返回右子树中含有的节点p或q;
        else if(left==NULL && right!=NULL) return right;
        // 如果左右子树都不包含,则返回空
        else return NULL;
    }
};

实际上我们写出的代码就已经包含了上面说的第二种情况。因为函数一开始就要判断当前节点是不是p或q,如果p或q一方本来就是最近公共祖先,那么他最后总会传递到最后。

  • 5
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
顺序存储的二叉树是一种利用数组来存储二叉树的方法,假设根节点存储在数组下标为1的位置,那么一个节点的左子节点的下标为2i,右子节点的下标为2i+1。而最近的公共祖先是指在一个二叉树中,节点p和q的最近公共祖先表示为一个节点x,满足x是p和q的祖先且x的深度尽可能大(一个节点也可以是它自己的祖先)。 对于顺序存储的二叉树,可以根据节点的下标来确定节点的父节点。如果节点i的下标不为1,则其父节点的下标为i/2。因此,可以从节点p和q开始,向上遍历找到它们的祖先节点,直到找到它们的公共祖先为止。 具体的算法如下: 1. 分别从节点p和q开始,向上遍历,将它们的祖先节点存储到两个数组中,直到根节点为止。 2. 从两个数组的末尾开始向前遍历,找到最后一个相同的节点,即为p和q的最近公共祖先。 代码实现如下: ```python def findCommonAncestor(tree, p, q): # 存储p和q的祖先节点 ancestors_p = [] ancestors_q = [] # 找到p和q的祖先节点 while p > 1: p //= 2 ancestors_p.append(p) while q > 1: q //= 2 ancestors_q.append(q) # 找到最近的公共祖先 common_ancestor = None while ancestors_p and ancestors_q and ancestors_p[-1] == ancestors_q[-1]: common_ancestor = ancestors_p.pop() ancestors_q.pop() return common_ancestor ``` 其中,tree是存储二叉树的数组,p和q是要找最近公共祖先的两个节点的下标。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值