二叉搜索树的最近公共祖先(leetcode)

我们来复习一下二叉搜索树(BST)的性质:

节点 N左子树上的所有节点的值都小于等于节点 N 的值
节点 N右子树上的所有节点的值都大于等于节点 N 的值
左子树和右子树也都是 BST

方法一 (递归)

思路

节点p,q 的最近公共祖先(LCA)是距离这两个节点最近的公共祖先节点。在这里 最近 考虑的是节点的深度。下面这张图能帮助你更好的理解 最近 这个词的含义。
二叉搜索树的最近公共祖先(leetcode)_第1张图片
笔记:p 和 q 其中的一个在 LCA 节点的左子树上,另一个在 LCA 节点的右子树上。

也有可能是下面这种情况:
二叉搜索树的最近公共祖先(leetcode)_第2张图片

算法

1,从根节点开始遍历树
2,如果节点 p 和节点 q 都在右子树上,那么以右孩子为根节点继续 1 的操作
3,如果节点 p 和节点 q 都在左子树上,那么以左孩子为根节点继续 1 的操作
4,如果条件 2 和条件 3 都不成立,这就意味着我们已经找到节 p 和节点 ,q 的 LCA 了

Python实现:

class Solution {
    public TreeNode lowestCommonAncestor(TreeNode root, TreeNode p, TreeNode q) {

        // Value of current node or parent node.
        int parentVal = root.val;

        // Value of p
        int pVal = p.val;

        // Value of q;
        int qVal = q.val;

        if (pVal > parentVal && qVal > parentVal) {
            // If both p and q are greater than parent
            return lowestCommonAncestor(root.right, p, q);
        } else if (pVal < parentVal && qVal < parentVal) {
            // If both p and q are lesser than parent
            return lowestCommonAncestor(root.left, p, q);
        } else {
            // We have found the split point, i.e. the LCA node.
            return root;
        }
    }
}

复杂度分析

时间复杂度:O(N) 其中 N 为 BST 中节点的个数,在最坏的情况下我们可能需要访问 BST 中所有的节点。

空间复杂度:O(N) 所需开辟的额外空间主要是递归栈产生的,之所以是 N 是因为 BST 的高度为 N。

方法二 (迭代)

算法

这个方法跟方法一很接近。唯一的不同是,我们用迭代的方式替代了递归来遍历整棵树。由于我们不需要回溯来找到 LCA 节点,所以我们是完全可以不利用栈或者是递归的。实际上这个问题本身就是可以迭代的,我们只需要找到分割点就可以了。这个分割点就是能让节点 p 和节点 q 不能在同一颗子树上的那个节点,或者是节点 p 和节点 q 中的一个,这种情况下其中一个节点是另一个节点的父亲节点。

Python实现:

class Solution:
    def lowestCommonAncestor(self, root, p, q):
        """
        :type root: TreeNode
        :type p: TreeNode
        :type q: TreeNode
        :rtype: TreeNode
        """

        # Value of p
        p_val = p.val

        # Value of q
        q_val = q.val

        # Start from the root node of the tree
        node = root

        # Traverse the tree
        while node:

            # Value of current node or parent node.
            parent_val = node.val

            if p_val > parent_val and q_val > parent_val:    
                # If both p and q are greater than parent
                node = node.right
            elif p_val < parent_val and q_val < parent_val:
                # If both p and q are lesser than parent
                node = node.left
            else:
                # We have found the split point, i.e. the LCA node.
                return node


复杂度分析

时间复杂度:O(N) 其中 N 为 BST 中节点的个数,在最坏的情况下我们可能需要遍历 BST 中所有的节点。

空间复杂度:O(1)

作者:LeetCode
链接:https://leetcode-cn.com/problems/two-sum/solution/er-cha-sou-suo-shu-de-zui-jin-gong-gong-zu-xian--2/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

你可能感兴趣的:(leetcode,Python)