【Day-35慢就是快】代码随想录-二叉树-二叉搜索树的最近公共祖先

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

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

思路


前面的二叉树思路为:利用回溯从底向上搜索,遇到一个节点的左子树里有p,右子树里有q,那么当前节点就是最近公共祖先。

在有序树里,如果判断一个节点的左子树里有p,右子树里有q呢?

因为是有序树,所有 如果 中间节点是 q 和 p 的公共祖先,那么 中节点的数组 一定是在 [p, q]区间的。即 中节点 > p && 中节点 < q 或者 中节点 > q && 中节点 < p。

那么只要从上到下去遍历,遇到 cur节点是数值在[p, q]区间中则一定可以说明该节点cur就是q 和 p的公共祖先。 那问题来了,一定是最近公共祖先吗

是的,第一次遇到的cur节点在区间[p, q]之间的,就是最近公共祖先。

递归法

1. 确定递归参数及返回值

参数为当前节点,以及两个P, Q;

返回值为最近公共祖先节点。

2. 终止条件

遇到空即可返回

3. 单层递归逻辑

在遍历二叉搜索树的时候就是寻找区间[p->val, q->val](注意这里是左闭又闭)

那么如果 cur->val 大于 p->val,同时 cur->val 大于q->val,那么就应该向左遍历(说明目标区间在左子树上)。

需要注意的是此时不知道p和q谁大,所以两个都要判断

if (cur->val > p->val && cur->val > q->val) {
    TreeNode* left = traversal(cur->left, p, q);
    if (left != NULL) {
        return left;
    }
}

整体代码:

class Solution {
public:
    TreeNode* traversal(TreeNode* cur, TreeNode* p, TreeNode* q){
        if(cur == NULL) return cur;

        if(cur->val > p->val && cur->val > q->val){
            TreeNode* left = traversal(cur->left,p, q);
            if(left != NULL) return left;
        }
        if(cur->val < p->val && cur->val < q->val){
            TreeNode* right = traversal(cur->right,p, q);
            if(right != NULL) return right;
        }
        return cur;
    }

    TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {
        return traversal(root, p, q);
    }
};

迭代法

class Solution {
public:
    TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {
        while(root){
            if(root->val > p->val && root->val >q->val){
                root = root->left;
            }else if(root->val < p->val && root->val < q->val)
                root = root->right;
            else return root;
        }
        return NULL;
    }
};

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
二叉搜索树(Binary Search Tree,简称BST)是一种特殊的二叉树,它的每个节点的值都大于其左子树中的所有节点的值,且小于其右子树中的所有节点的值。最近公共祖先(Lowest Common Ancestor,简称LCA)是指在二叉树中,两个节点p和q的最近公共祖先节点。 对于给定的二叉搜索树,我们可以通过比较节点的值来确定最近公共祖先节点。具体步骤如下: 1. 从根节点开始遍历二叉搜索树。 2. 如果当前节点的值大于p和q的值,说明p和q都在当前节点的左子树中,因此继续遍历当前节点的左子树。 3. 如果当前节点的值小于p和q的值,说明p和q都在当前节点的右子树中,因此继续遍历当前节点的右子树。 4. 如果当前节点的值介于p和q的值之间,说明当前节点就是最近公共祖先节点。 以下是一个示例代码,演示了如何找到二叉搜索树最近公共祖先节点: ```python class TreeNode: def __init__(self, val=0, left=None, right=None): self.val = val self.left = left self.right = right def lowestCommonAncestor(root, p, q): if root.val > p.val and root.val > q.val: return lowestCommonAncestor(root.left, p, q) elif root.val < p.val and root.val < q.val: return lowestCommonAncestor(root.right, p, q) else: return root # 创建一个二叉搜索树 root = TreeNode(6) root.left = TreeNode(2) root.right = TreeNode(8) root.left.left = TreeNode(0) root.left.right = TreeNode(4) root.right.left = TreeNode(7) root.right.right = TreeNode(9) root.left.right.left = TreeNode(3) root.left.right.right = TreeNode(5) # 找到节点3和节点5的最近公共祖先 p = TreeNode(3) q = TreeNode(5) lca = lowestCommonAncestor(root, p, q) print("最近公共祖先节点的值为:", lca.val) # 输出:2 ```

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值