代码随想录算法训练营第十五天|二叉树的层序遍历、226.翻转二叉树、101.对称二叉树

代码随想录算法训练营第十五天|226.翻转二叉树、101.对称二叉树

层序遍历:(10个相关题目)

题解:用队列来保存每一层遍历过的元素,将每一层的元素放在一个list里面,最后将每一层的list加入到结果list里面。每一层遍历的时候,除了要将自己的值放入队列中,还要将自己的孩子放进队列中,作为下一层遍历。

List<List<Integer>> res=new ArrayList<List<Integer>>();
Queue<TreeNode> que=new LinkedList<>();
que.offer(root);
while(!que.isEmpty()){
    List<Integer> temp=new ArrayList<>();
    int len=que.size();
    while(len>0){
        TreeNode node=que.poll();
        temp.add(node.val);
        if(node.left!=null) que.offer(node.left);
        if(node.right!=null) que.offer(node.right);
        len--;
    }
    res.add(temp);
}

102.二叉树的层序遍历

给你二叉树的根节点 root ,返回其节点值的 层序遍历 。 (即逐层地,从左到右访问所有节点)。

示例 1:

img

输入:root = [3,9,20,null,null,15,7]
输出:[[3],[9,20],[15,7]]

代码

class Solution {
    public List<List<Integer>> levelOrder(TreeNode root) {
        List<List<Integer>> res=new ArrayList<>();
        if(root==null) return res;
        Queue<TreeNode> que=new LinkedList<>();
        que.offer(root);
        while(!que.isEmpty()){
            int len=que.size();
            List<Integer> temp=new ArrayList<>();
            while(len>0){
                TreeNode node=que.poll();
                temp.add(node.val);

                if(node.left!=null) que.offer(node.left);
                if(node.right!=null) que.offer(node.right);
                len--;
            }
            res.add(temp);
        }
        return res;
    }
}

107.二叉树的层序遍历II

给你二叉树的根节点 root ,返回其节点值 自底向上的层序遍历 。 (即按从叶子节点所在层到根节点所在的层,逐层从左向右遍历)

题解:就是层序遍历,然后结果数组反转一下就可以。

class Solution {
    public List<List<Integer>> levelOrderBottom(TreeNode root) {
        List<List<Integer>> res=new ArrayList<>();
        if(root==null) return res;
        Queue<TreeNode> que=new LinkedList<>();
        que.offer(root);
        while(!que.isEmpty()){
            int len=que.size();
            List<Integer> temp=new ArrayList<>();
            while(len>0){
                TreeNode node=que.poll();
                temp.add(node.val);

                if(node.left!=null) que.offer(node.left);
                if(node.right!=null) que.offer(node.right);
                len--;
            }
            res.add(temp);
        }
        Collections.reverse(res);
        return res;
    }
}

199.二叉树的右视图

给定一个二叉树的 根节点 root,想象自己站在它的右侧,按照从顶部到底部的顺序,返回从右侧所能看到的节点值。

示例 1:

img

输入: [1,2,3,null,5,null,4]
输出: [1,3,4]

题解:只加右边的节点即可。需要注意的是,最右边的节点可以是左子树的右节点(当左子树比右子树长时)

代码

class Solution {
    public List<Integer> rightSideView(TreeNode root) {
        List<Integer> res=new ArrayList<>();
        if(root==null) return res;
        Queue<TreeNode> que=new LinkedList<>();
        que.offer(root);
        while(!que.isEmpty()){
            int len=que.size();
            for(int i=0;i<len;i++){
                TreeNode node=que.poll();
                if(node.left!=null) que.offer(node.left);
                if(node.right!=null) que.offer(node.right);
                if(i==len-1) res.add(node.val);
            }
        }
        return res;
    }
}

637.二叉树的层平均值

给定一个非空二叉树的根节点 root , 以数组的形式返回每一层节点的平均值。与实际答案相差 10-5 以内的答案可以被接受。

示例 1:

img

输入:root = [3,9,20,null,null,15,7]
输出:[3.00000,14.50000,11.00000]
解释:第 0 层的平均值为 3,第 1 层的平均值为 14.5,第 2 层的平均值为 11 。
因此返回 [3, 14.5, 11] 。

题解:一样的思路。就是将每一层的值加起来,最后除以每一层的元素个数,将这个值加入到结果数组就可以了。

代码

class Solution {
    public List<Double> averageOfLevels(TreeNode root) {
        List<Double> res=new ArrayList<>();
        if(root==null) return res;
        Queue<TreeNode> que=new LinkedList<>();
        que.offer(root);
        while(!que.isEmpty()){
            int len=que.size();
            int a=len;
            double sum=0;
            while(len>0){
                TreeNode node=que.poll();
                sum+=node.val;

                if(node.left!=null) que.offer(node.left);
                if(node.right!=null) que.offer(node.right);
                len--;
            }

            res.add(sum/a);
        }
        return res;
    }
}

429.N叉树的层序遍历

给定一个 N 叉树,返回其节点值的层序遍历。(即从左到右,逐层遍历)。

树的序列化输入是用层序遍历,每组子节点都由 null 值分隔(参见示例)。

示例 1:

img

输入:root = [1,null,3,2,4,null,5,6]
输出:[[1],[3,2,4],[5,6]]

题解这个题一开始没有思路:怎么去找一个节点的多个孩子呢?。原来可以直接List chilsren=node.children。

一个节点有多个孩子,可以直接通过.children的方式将孩子都取出来。

代码

class Solution {
    public List<List<Integer>> levelOrder(Node root) {
        List<List<Integer>> res=new ArrayList<>();
        if(root == null) return res;
        Queue<Node>  que=new LinkedList<>();
        que.offer(root);
        while(!que.isEmpty()){
            List<Integer> list=new ArrayList<>();
            int len=que.size();
            for(int i=0;i<len;i++){
                Node node=que.poll();
                list.add(node.val);
                List<Node> cnode=node.children;
                if(cnode==null || cnode.size()==0){
                    continue;
                }
                for(Node n:cnode){
                    if(n!=null){
                        que.offer(n);
                    }
                }
            }
            res.add(list);
        }
        return res;
    }
}

515.在每个树中找最大值

给定一棵二叉树的根节点 root ,请找出该二叉树中每一层的最大值。

示例1:

img

输入: root = [1,3,2,5,3,null,9]
输出: [1,3,9]

题解:层序遍历,在将每层的节点值都加入list后,找出最大值就可以了。可以单独写一个找出最大值的函数,也可以直接使用库函数 max=Math.max(值1,值2)。

代码

class Solution {
    public List<Integer> largestValues(TreeNode root) {
        List<Integer> res=new ArrayList<>();
        if(root==null) return res;
        Queue<TreeNode> que=new LinkedList<>();
        que.offer(root);
        while(!que.isEmpty()){
            int len=que.size();
            List<Integer> list=new ArrayList<>();
            while(len>0){
                TreeNode node=que.poll();
                list.add(node.val);
                if(node.left!=null) que.offer(node.left);
                if(node.right!=null) que.offer(node.right);
                len--;
            }
            int ans=big(list);
            res.add(ans);
        }
        return res;
     }
    public int big(List<Integer> list){
        int max=Integer.MIN_VALUE;
        for(int n:list){
            max=max>n?max:n;
        }
        return max;
    }
}

116.填充每个节点的下一右侧指针节点

给定一个 完美二叉树 ,其所有叶子节点都在同一层,每个父节点都有两个子节点。二叉树定义如下:

struct Node {
  int val;
  Node *left;
  Node *right;
  Node *next;
}

填充它的每个 next 指针,让这个指针指向其下一个右侧节点。如果找不到下一个右侧节点,则将 next 指针设置为 NULL

初始状态下,所有 next 指针都被设置为 NULL

示例 1:

img

输入:root = [1,2,3,4,5,6,7]
输出:[1,#,2,3,#,4,5,6,7,#]
解释:给定二叉树如图 A 所示,你的函数应该填充它的每个 next 指针,以指向其下一个右侧节点,如图 B 所示。序列化的输出按层序遍历排列,同一层节点由 next 指针连接,'#' 标志着每一层的结束。

题解:层序遍历。将当前节点拿出来后,它的next节点就是队列中的第一个元素,注意处理每一层的最后一个节点就可以了。

代码

class Solution {
    public Node connect(Node root) {
        if(root==null) return null;
        Queue<Node> que=new LinkedList<>();
        que.offer(root);
        while(!que.isEmpty()){
            int len=que.size();
            for(int i=0;i<len;i++){
                Node node=que.poll();
                if(i<len-1){
                    node.next=que.peek();
                }else{
                    //最后一个节点让它指向null
                    node.next=null;
                }
                if(node.left!=null) que.offer(node.left);
                if(node.right!=null) que.offer(node.right);
            }
        }
        return root;
    }
}

117.填充每个节点的下一右侧指针节点II

给定一个二叉树:

struct Node {
  int val;
  Node *left;
  Node *right;
  Node *next;
}

填充它的每个 next 指针,让这个指针指向其下一个右侧节点。如果找不到下一个右侧节点,则将 next 指针设置为 NULL

初始状态下,所有 next 指针都被设置为 NULL

示例 1:

img

输入:root = [1,2,3,4,5,null,7]
输出:[1,#,2,3,#,4,5,7,#]
解释:给定二叉树如图 A 所示,你的函数应该填充它的每个 next 指针,以指向其下一个右侧节点,如图 B 所示。序列化输出按层序遍历顺序(由 next 指针连接),'#' 表示每层的末尾。

题解:和上一题的解法是一样的。

代码

class Solution {
    public Node connect(Node root) {
        if(root == null)  return null;
        Queue<Node> que=new LinkedList<>();
        que.offer(root);
        while(!que.isEmpty()){
            int len=que.size();
            for(int i=0;i<len;i++){
                Node node=que.poll();
                if(i!=len-1){
                    node.next=que.peek();
                }else{
                    node.next=null;
                }
                if(node.left!=null)  que.offer(node.left);
                if(node.right!=null)  que.offer(node.right);
            }
        }
        return root;
    }
}

104.二叉树的最大深度

给定一个二叉树 root ,返回其最大深度。

二叉树的 最大深度 是指从根节点到最远叶子节点的最长路径上的节点数。

示例 1:

img

输入:root = [3,9,20,null,null,15,7]
输出:3

题解:层序遍历,遍历一层树的深度加一。

代码

class Solution {
    public int maxDepth(TreeNode root) {
        int res=0;
        if(root==null) return res;
        Queue<TreeNode> que=new LinkedList<>();
        que.offer(root);
        while(!que.isEmpty()){
            int len=que.size();
            while(len>0){
                TreeNode node=que.poll();
                if(node.left!=null) que.offer(node.left);
                if(node.right!=null) que.offer(node.right);
                len--;
            }
            res+=1;
        }
        return res;
    }
}

111.二叉树的最小深度

给定一个二叉树,找出其最小深度。

最小深度是从根节点到最近叶子节点的最短路径上的节点数量。

**说明:**叶子节点是指没有子节点的节点。

示例 1:

img

题解:层序遍历。需要注意的是,当遇到第一个叶子节点后后就直接返回,不要break,不然它只退出了内层循环,外层循环还在继续

代码

class Solution {
    public int minDepth(TreeNode root) {
        int res=0;
        if(root==null) return res;
        Queue<TreeNode> que=new LinkedList<>();
        que.offer(root);
        while(!que.isEmpty()){
            int len=que.size();
            res+=1;
            while(len>0){
                TreeNode node=que.poll();
                if(node.left==null && node.right==null) {
                   return res;
                }
                if(node.left!=null) que.offer(node.left);
                if(node.right!=null) que.offer(node.right);
                len--;
            }
        }
        return res;
    }
}

226.翻转二叉树

给你一棵二叉树的根节点 root ,翻转这棵二叉树,并返回其根节点。

示例 1:

img

输入:root = [4,2,7,1,3,6,9]
输出:[4,7,2,9,6,3,1]

示例 2:

img

输入:root = [2,1,3]
输出:[2,3,1]

题解:将节点加入队列中,然后交换它的左右孩子节点,记录每一层的节点个数,然后每个节点的孩子都要交换。注意要交换的是当前节点的孩子节点,然后再次遍历,再次交换

代码

class Solution {
    public TreeNode invertTree(TreeNode root) {
        if(root==null) return null;
        Queue<TreeNode> que=new LinkedList<>();
        que.offer(root);
        while(!que.isEmpty()){
            int len=que.size();
            while(len>0){
                TreeNode node=que.poll();
                swap(node);
                if(node.left!=null) que.offer(node.left);
                if(node.right!=null) que.offer(node.right);
                len--;
            }
        }
        return root;
    }
    public void swap(TreeNode node){
        TreeNode temp=node.left;
        node.left=node.right;
        node.right=temp;
    }
}

101.对称二叉树

给你一个二叉树的根节点 root , 检查它是否轴对称。

示例 1:

img

输入:root = [1,2,2,3,4,4,3]
输出:true

示例 2:

img

输入:root = [1,2,2,null,3,null,3]
输出:false

题解:比较左右子树是否相等,那么就分别比较节点的左右孩子是否相等,这里入队列的顺序要注意一下,因为是最左边与最右边比较。(看着代码理解)

代码

class Solution {
    public boolean isSymmetric(TreeNode root) {
        Queue<TreeNode> que=new LinkedList<>();
        que.offer(root.left);
        que.offer(root.right);
        while(!que.isEmpty()){
            TreeNode lnode=que.poll();
            TreeNode rnode=que.poll();
            if(lnode == null &&rnode == null){
               continue;
            }
            if(lnode==null || rnode==null ||lnode.val!=rnode.val){
                return false;
            }
            que.offer(lnode.left);
            que.offer(rnode.right);
            que.offer(lnode.right);
            que.offer(rnode.left);
        }
        return true;
    }
}

100.相同的树

给你两棵二叉树的根节点 pq ,编写一个函数来检验这两棵树是否相同。

如果两个树在结构上相同,并且节点具有相同的值,则认为它们是相同的。

示例 1:

img

输入:p = [1,2,3], q = [1,2,3]
输出:true

题解注意是两棵树的比较,左节点和左节点比较,,不要当成镜像

代码

class Solution {
    public boolean isSameTree(TreeNode p, TreeNode q) {
        if(p==null &&q==null)  return true;
        if(p==null || q==null) return false;
        Queue<TreeNode> que=new LinkedList<>();
        que.offer(p);
        que.offer(q);
        while(!que.isEmpty()){
            TreeNode node1=que.poll();
            TreeNode node2=que.poll();
            if(node1.val!=node2.val){
                return false;
            }
            if(node1.left!=null&&node2.left==null || node1.left==null && node2.left!=null){
                return false;
            }
            if(node1.right!=null&&node2.right==null || node1.right==null && node2.right!=null){
                return false;
            }
            if(node1.left!=null) que.offer(node1.left);
            if(node2.left!=null) que.offer(node2.left);
            if(node1.right!=null) que.offer(node1.right);
            if(node2.right!=null) que.offer(node2.right);
        }
        return true;
    }
}

572.另一个树的子树

给你两棵二叉树 rootsubRoot 。检验 root 中是否包含和 subRoot 具有相同结构和节点值的子树。如果存在,返回 true ;否则,返回 false

二叉树 tree 的一棵子树包括 tree 的某个节点和这个节点的所有后代节点。tree 也可以看做它自身的一棵子树。

示例 1:

img

输入:root = [3,4,5,1,2], subRoot = [4,1,2]
输出:true

题解:两重递归。一重判断是不是子树,另一重递归判断两棵树是否相等。

代码

class Solution {
    //判断是不是子树
    public boolean isSubtree(TreeNode root, TreeNode subRoot) {
        if(root==null) return false;
        if(subRoot==null) return true;

        return digui(root,subRoot)||isSubtree(root.left,subRoot)||isSubtree(root.right,subRoot);
    }
    //递归判断两棵树是否相等
    public boolean digui(TreeNode s,TreeNode t){
        if(s==null && t==null) return true;
        if(s==null || t==null) return false;
        if(s.val!=t.val) return false;
        return digui(s.left,t.left)&&digui(s.right,t.right);
    }
}
  • 4
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值