代码随想录算法训练营第十七天/110.平衡二叉树、257.二叉树的所有路径、404.左叶子之和


110.平衡二叉树

思路

平衡二叉树概念:左右子树高度差不大于1;
首先当节点为空时,返回的高度为0;利用递归先计算出当前节点的左右子树高度,将两高度相减并获得绝对值,若绝对值大于1则表示不是平衡二叉树,返回-1,最后返回两高度中最大的加1,这个是计算出最大的子树高度与另外子树高度相减;还有要判断通过递归计算出的左右子树高度,是存在值为-1的情况,表示不满足,那么就要返回-1;所有在主方法中返回调用方法!=-1,即为满足平衡二叉树,反之不满足返回false。

注意

(1)是计算高度,要用后序遍历

实现代码

class Solution {
    public boolean isBalanced(TreeNode root) {
        return getHigh(root)!=-1;        
    }
    public int getHigh(TreeNode root){
        if(root==null){
            return 0;
        }
        int highleft=getHigh(root.left);
        int highright=getHigh(root.right);
        if(highleft==-1){
            return -1;
        }
        if(highright==-1){
            return -1;
        }

        if(Math.abs(highleft-highright)>1){
            return -1;
        }

        //计算左右子树中最大高度
        return Math.max(highleft,highright)+1;
    }
}

257.二叉树的所有路径

思路

设置两个集合,一个是存放路径的,另外一个是存放遍历到的节点的值的;设置一个StringBuilder,当满足当前节点也就是节点左右子树为空时,就开始遍历第二个集合,将遍历到的值用append方法加入到StringBuilder中,然后再append("->),遍历一直到倒数第二个,因为有符号连接,在结束循环后再将最后一个元素append入StringBuilder中,最后将StringBuilder使用toString()方法加入到存放路径的集合中;接着要处理当前节点的左右子树,如果左右子树存在,那么就递归,并且回溯,这里的回溯是将存放节点值的集合最后一个元素给移除,回溯多少就从后面开始移除多少,左右子树均如此。

注意

为什么要回溯?
根据题目所示可以知道要获得所有的路径,有时候某个节点可能会有左右两边几条路径,所以需要回溯去找到另外的路径,那我每遍历一个节点就将它的值给加入到指定集合中,现在回溯等于要把值移除,回到某个位置,回溯多少就对应移除多少,在继续遍历另外一条路径,再继续将遍历新路径的节点值放入集合中。

实现代码

class Solution {
    public List<String> binaryTreePaths(TreeNode root) {
        List<String> res=new ArrayList<>();
        List<Integer> list=new ArrayList<>();
        getAll(root,res,list);
        return res;     
    }

    public void getAll(TreeNode root,List<String> res,List<Integer> list){
        list.add(root.val);
        if(root.left==null&&root.right==null){
            StringBuilder str=new StringBuilder();
            for(int i=0;i<list.size()-1;i++){
                str.append(list.get(i)).append("->");
            }
            str.append(list.get(list.size()-1));
            res.add(str.toString());
            return;
        }

        if(root.left!=null){
            getAll(root.left,res,list);
            list.remove(list.size()-1);//回溯
        }
        if(root.right!=null){
            getAll(root.right,res,list);
            list.remove(list.size()-1);//回溯
        }
    }
}

404.左叶子之和

思路

采用哪种遍历方式都可,此题不是直接利用叶子节点为当前节点,而是用到了它的父节点,通过判断父节点的左节点不为空,而父节点的左节点的左右节点都为空时,此父节点的左节点满足并且是左子叶,所有获得它的值;然后将左右子树中找到的左子叶的值与此获得值相加,将相加后的总和相加即是此树所有的左子叶之和。

注意

不同于以往是以当前节点作为满足要求的节点,此题是以满足的节点的父节点作为当前节点的!!!

实现代码

class Solution {
    public int sumOfLeftLeaves(TreeNode root) {
        int sum=0;
        if(root==null){
            return 0;
        }
        if(root.left!=null&&root.left.left==null&&root.left.right==null){
            sum=root.left.val;
        }
        int sumleft=sumOfLeftLeaves(root.left);
        int sumright=sumOfLeftLeaves(root.right);
        return sumleft+sumright+sum;
    }
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值