zoukankan      html  css  js  c++  java
  • LeetCode236-最早公共祖先(递归)

    递归去找这个结点,如果找到了,就将父元素加到list里面去。

    其实就是找到达那个节点的路劲。

    这个递归我觉得自己写的很好,巧妙的利用返回值true和false来判断是否将自己加进去。同时在参数里面传list去保存,非常方便。

    最后会有两个list,将这两个list,从右往左比较,应该都是相同的元素,第一个不同的元素的上一个元素,就是最先公共祖先。

    public class LeetCode236 {
    
        public static void main(String[] args) {
    
        }
    
        public static boolean find(TreeNode node, TreeNode target, LinkedList<TreeNode> queue){
    
            if(node==null)
                return false;
    
            if(node==target){
                //因为是包括自己的,自己也能算别人的祖先
                queue.add(node);
                return true;
            }
    
            //左边找到就不去找右边了
            //左右孩子如果能找到那个元素,自己就是祖先,把自己加进去
            boolean left = find(node.left,target,queue);
            if(left){
                queue.add(node);
                return true;
            }
    
            boolean right = find(node.right,target,queue);
            if(right){
                queue.add(node);
                return true;
            }
    
            //左右孩子都没有,那就找不到了
            return false;
    
    
        }
    
        public TreeNode lowestCommonAncestor(TreeNode root, TreeNode p, TreeNode q) {
    
            //获取到达p的路径
            LinkedList<TreeNode> list1 = new LinkedList<>();
            find(root,p,list1);
    
            LinkedList<TreeNode> list2 = new LinkedList<>();
            find(root,q,list2);
    
            TreeNode last = null;
    
            //从右边开始
            while (list1.size()!=0&&list2.size()!=0){
    
                //poll是返回队列头部
                TreeNode temp1 = list1.pollLast();
                TreeNode temp2 = list2.pollLast();
    
                if(temp1==temp2)
                    last=temp1;
                else
                    break;
    
            }
    
            return last;
    
    
        }
    
    }

    其实网上还有更简单的写法。

    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都找到了,因为p不可能又在左子树,又在右子树,所以如果一个是p,另一个肯定是q
            //如果pq在一个结点的左右两侧,那么这个就是最近的了,更高一级的父节点,pq是在一侧的
            if(left != null && right != null) {
                return root;
            }
            //如果只找到一个,就返回它自己,因为他代表了到达目标的路径,他能到达目标结点
    //在更高一级的结点的时候,就回子递归的结果,因为是最近父结点
    return left != null ? left : right; } }

    这个解法太巧妙了

    每一步,在不同的情况下,都发挥了不同的效果

    很难想到

    很好的解决了,p或者q本身就是公共结点的情况

  • 相关阅读:
    鼠标点击表格行背景变色
    2006年星座运势全解巨蟹
    去除衣物污渍大本营
    海量数据库的查询优化及分页算法方案[转帖]
    奇怪的Access错误
    深圳易高科技有限公司面试题目
    各大IT公司的起名缘由
    微星横向菜单
    【转】函数参数入栈问题
    堆和栈的区别 (转贴)
  • 原文地址:https://www.cnblogs.com/weizhibin1996/p/9694062.html
Copyright © 2011-2022 走看看