二叉树层级遍历以及相关练习

news2025/2/27 21:00:40

二叉树层级遍历以及相关练习

文章目录

  • 二叉树层级遍历以及相关练习
    • 思想
    • 步骤
    • 代码实现
    • 相关练习

力扣:102. 二叉树的层序遍历 - 力扣(Leetcode)

思想

层序遍历一个二叉树。就是从左到右一层一层的去遍历二叉树。

使用队列实现二叉树广度优先遍历,动画如下:

在这里插入图片描述

步骤

  1. 先将根节点放到队列中
  2. 记录当前队列元素个数
  3. 出队,将出对的元素接收起来
  4. 判断是否有左右节点,有就加入队列
  5. 重复2到4步骤,直到队列中没有元素

代码实现

package com.algo.tree.binarytree;

import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;
import java.util.Queue;

public class Algo102 {
    //二叉树的层级遍历
    public List<List<Integer>> levelOrder(TreeNode root) {
        List<List<Integer>> lists = new ArrayList<>();
        if(root==null){
            return lists;
        }
        Queue<TreeNode> queue = new LinkedList<>();
        //先将根节点放到队列中
        queue.add(root);
        while (!queue.isEmpty()){
            //记录当前层的元素个数
            int num = queue.size();
            List<Integer> list = new ArrayList<>();
            //从队列中依次取出num个元素进行出队  即到哪一层就将哪一层元素全部出队
            for (int i = 0; i < num; i++) {
                //出队
                TreeNode node = queue.remove();
                if(node.left!=null){//如果当前节点的左子节点不为空,那么就将当前节点的左子节点入队
                    queue.add(node.left);
                }
                if(node.right!=null){//如果当前节点的右子节点不为空,那么就将当前节点的右子节点入队
                    queue.add(node.right);
                }
                //将当前节点的值放到集合中
                list.add(node.val);
            }
            lists.add(list);
        }
        return lists;
    }

}

相关练习

  • 107.二叉树的层次遍历II

力扣:107. 二叉树的层序遍历 II - 力扣(Leetcode)

  1. 先将二叉树层级遍历
  2. 然后将每层的元素保存到每个集合中
  3. 再将这些结合翻转

代码实现

package com.algo.tree.binarytree;

import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;
import java.util.Queue;

public class Algo107 {
    public List<List<Integer>> levelOrderBottom(TreeNode root) {
        List<List<Integer>> lists = new ArrayList<>();
        if (root == null) {
            return lists;
        }
        Queue<TreeNode> queue = new LinkedList<>();
        queue.add(root);
        //层级遍历
        while (!queue.isEmpty()) {
            int num = queue.size();
            List<Integer> list = new ArrayList<>();
            for (int i = 0; i < num; i++) {
                TreeNode node = queue.remove();
                if (node.left != null) {
                    queue.add(node.left);
                }
                if (node.right != null) {
                    queue.add(node.right);
                }
                //将每层元素添加到列表中
                list.add(node.val);
            }
            //将每个集合添加到列表中
            lists.add(list);
        }
        //将lists中的元素位置互换
        int left = 0;
        int right = lists.size() - 1;
        while (left <= right) {
            List<Integer> list = lists.get(left);
            lists.set(left, lists.get(right));
            lists.set(right, list);
            left++;
            right--;
        }
        return lists;
    }
}

  • 199.二叉树的右视图

力扣:199. 二叉树的右视图 - 力扣(Leetcode)

  1. 右视图无非就是每层的最后一个元素
  2. 先层级遍历二叉树
  3. 当碰到每层的最后一个元素就加入到列表中

代码实现

package com.algo.tree.binarytree;

import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;
import java.util.Queue;

public class Algo199 {
    public List<Integer> rightSideView(TreeNode root) {
        List<Integer> list = new ArrayList<>();
        if(root==null){
            return list;
        }
        Queue<TreeNode> queue = new LinkedList<>();
        queue.add(root);
        //层级遍历
        while (!queue.isEmpty()){
            int num = queue.size();
            for (int i = 0; i < num; i++) {
                TreeNode node = queue.remove();
                if(node.left!=null){
                    queue.add(node.left);
                }
                if(node.right!=null){
                    queue.add(node.right);
                }
                //将每层的最后一个元素加入到list中
                if(i==num-1){
                    list.add(node.val);
                }
            }
        }
        return list;
    }
}

  • 637.二叉树的层平均值

力扣:637. 二叉树的层平均值 - 力扣(Leetcode)

  1. 每层在遍历之前都创建一个sum
  2. 遍历一层中的每个元素都加到sum中
  3. 在遍历之后取平均值
  4. 将平均值加到列表中

代码实现

package com.algo.tree.binarytree;

import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;
import java.util.Queue;

public class Algo637 {
    public List<Double> averageOfLevels(TreeNode root) {
        List<Double> list = new ArrayList<>();
        if(root==null){
            return list;
        }
        Queue<TreeNode> queue = new LinkedList<>();
        queue.add(root);
        while (!queue.isEmpty()){
            double num = queue.size();
            //遍历之前先创建sum
            double sum = 0;
            for (int i = 0; i < num; i++) {
                TreeNode node = queue.remove();
                if(node.left!=null){
                    queue.add(node.left);
                }
                if(node.right!=null){
                    queue.add(node.right);
                }
                //将该层的元素的值全部相加
                sum+=node.val;
            }
            //取平均值加到列表中
            list.add( sum/num);
        }
        return list;
    }
}

  • 429.N叉树的层序遍历

力扣:429. N 叉树的层序遍历 - 力扣(Leetcode)

  1. N叉树的层级遍历步骤和二叉树的层级遍历步骤几乎一样
  2. 先将根节点放到队列中
  3. 再记录当前队列元素个数
  4. 出队,将出队元素保存
  5. 如果该元素有子节点列表,那么就将子节点列表中的元素依次加入到队列中
  6. 直到队列为空,遍历完毕
package com.algo.tree.binarytree;

import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;
import java.util.Queue;

public class Algo429 {
    //N 叉树的层序遍历
    public List<List<Integer>> levelOrder(Node root) {
        List<List<Integer>> lists = new ArrayList<>();
        if(root==null){
            return lists;
        }
        Queue<Node> queue = new LinkedList<>();
        queue.add(root);
        //层级遍历
        while (!queue.isEmpty()){
            //记录当前队列元素个数
            int num = queue.size();
            List<Integer> list = new ArrayList<>();
            for (int i = 0; i < num; i++) {
                Node node = queue.poll();
                if(node.children!=null){
                    //将当前节点的所有子节点都放到队列中
                    for(Node node1:node.children){
                        queue.add(node1);
                    }
                }
                //将当前节点的值放到集合中
                list.add(node.val);
            }
            //将当前的集合放到lists中
            lists.add(list);
        }
        return lists;
    }
}

  • 515.在每个树行中找最大值

力扣:515. 在每个树行中找最大值 - 力扣(Leetcode)

  1. 在遍历每一层之前创建一个数组
  2. 用该数组保存该层的每一个元素
  3. 当该层遍历完毕就对数组进行从小到大排序
  4. 然后将该数组的最后一个元素加入到列表中
package com.algo.tree.binarytree;

import java.util.*;

public class Algo515 {
    public List<Integer> largestValues(TreeNode root) {
        List<Integer> list = new ArrayList<>();
        if(root==null){
            return list;
        }
        Queue<TreeNode> queue = new LinkedList<>();
        queue.add(root);
        while (!queue.isEmpty()){
            int num = queue.size();
            int[] arr = new int[num];
            //遍历每一层
            for (int i = 0; i < num; i++) {
                TreeNode node = queue.poll();
                if(node.left!=null){
                    queue.add(node.left);
                }
                if(node.right!=null){
                    queue.add(node.right);
                }
                //将每个值存储起来
                arr[i] = node.val;
            }
            //排序
            Arrays.sort(arr);
            //将排完序的数组的最后一位放到list中
            list.add(arr[num-1]);
        }
        return list;
    }
}

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

力扣:116. 填充每个节点的下一个右侧节点指针 - 力扣(Leetcode)

  1. 每次递归前都将当前节点的下一层处理完毕
  2. 如果当前节点的左子节点不为空 让当前节点的左子节点的next指向当前节点的右子节点
  3. 如果当前节点的next不为空 让当前节点的右子节点的next指向下一个节点的左子节点
  4. 向左递归
  5. 向右递归
package com.algo.tree.binarytree;

import java.util.LinkedList;
import java.util.Queue;

public class Algo116 {
    public Node1 connect(Node1 root) {
        if(root==null){
            return root;
        }
        //每次递归前都将当前节点的下一层处理完毕
        //如果当前节点的左子节点不为空 让当前节点的左子节点的next指向当前节点的右子节点
        if(root.left!=null){
            root.left.next = root.right;
            if(root.next!=null){//如果当前节点的next不为空 让当前节点的右子节点的next指向下一个节点的左子节点
                root.right.next = root.next.left;
            }
        }
        connect(root.left);
        connect(root.right);
        return root;
    }
}


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

力扣:117. 填充每个节点的下一个右侧节点指针 II - 力扣(Leetcode)

  1. 因为这不是完美二叉树,所以得用层级遍历的思想
  2. 遍历每一层,在未遍历完该层时让出队列的元素的next指向队列头部元素
  3. 当遍历到该层最后一个元素时,就将该元素的next指向null
  4. 最后返回头结点
package com.algo.tree.binarytree;


import java.util.LinkedList;
import java.util.Queue;

public class Algo117 {
    public Node1 connect(Node1 root) {
        if(root==null){
            return root;
        }
        Queue<Node1> queue = new LinkedList<>();
        queue.add(root);
        //层级遍历
        while (!queue.isEmpty()){
            int num = queue.size();
            for (int i = 0; i < num; i++) {
                Node1 node1 = queue.poll();
                if(node1.left!=null){
                    queue.add(node1.left);
                }
                if(node1.right!=null){
                    queue.add(node1.right);
                }
                if(i==num-1){
                    //让当前层的最后一个元素指向null
                    node1.next=null;
                }else{
                    //让当前节点指向队列的头  当当前层的元素没有出完时 对头元素永远是当前节点的下一个元素
                    node1.next = queue.peek();
                }
            }
        }
        return root;
    }
}


  • 104.二叉树的最大深度

力扣:104. 二叉树的最大深度 - 力扣(Leetcode)

  1. 分别向左和向右递归
  2. 返回深度最大的值
package com.algo.tree.binarytree;


public class Algo104 {
    public int maxDepth(TreeNode root) {
        return root == null ? 0 : Math.max(maxDepth(root.left), maxDepth(root.right)) + 1;
    }
}

  • 111.二叉树的最小深度

力扣:111. 二叉树的最小深度 - 力扣(Leetcode)

  1. 有5中情况
  2. 第一种:当根节点为空时返回0
  3. 第二种:当根节点不为空但是没有子节点返回1
  4. 第三种:当根节点不为空且左右子节点也不为空时,分别向左和向右递归,取最小值
  5. 第四种:当根节点不为空且左子节点不为空而右子节点为空,那么向左子节点递归
  6. 第五种:当根节点不为空且右子节点不为空而左子节点为空,那么向右子节点递归
package com.algo.tree.binarytree;

public class Algo111 {
    public int minDepth(TreeNode root) {
        if(root==null){
            return 0;
        }else if(root.left==null&&root.right==null){
            return 1;
        }else if(root.left!=null&&root.right!=null){
            return Math.min(minDepth(root.left),minDepth(root.right))+1;
        }else if(root.left!=null){
            return minDepth(root.left)+1;
        }else {
            return minDepth(root.right)+1;
        }
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/520423.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

算法修炼之练气篇——练气二十层

博主&#xff1a;命运之光 专栏&#xff1a;算法修炼之练气篇 前言&#xff1a;每天练习五道题&#xff0c;炼气篇大概会练习200道题左右&#xff0c;题目有C语言网上的题&#xff0c;也有洛谷上面的题&#xff0c;题目简单适合新手入门。&#xff08;代码都是命运之光自己写的…

实例演示如何结合Selenium和Requests进行自动化测试

下方查看历史精选文章 重磅发布 - 自动化框架基础指南pdfv1.1大数据测试过程、策略及挑战 测试框架原理&#xff0c;构建成功的基石 在自动化测试工作之前&#xff0c;你应该知道的10条建议 在自动化测试中&#xff0c;重要的不是工具 Selenium和Requests是两个常用的自动化测试…

Meme和BRC-20的暴跌洗盘

* * * 原创&#xff1a;刘教链 * * * 5月以来的Meme&#xff08;模因币&#xff0c;发音类似“谜姆”&#xff09;狂欢&#xff0c;有ERC-20版本的Meme比如Pepe&#xff08;青蛙佩佩&#xff09;&#xff0c;也有BRC-20版本的Meme比如ordi&#xff08;序数协议本身的名字&…

IM即时通讯系统[SpringBoot+Netty]——梳理(二)

文章目录 五、IM开发核心之构建TCP网关&#xff08;上&#xff09;1、编写LimServer2、编写LimWebSocketServer3、使用snakeyaml动态配置文件4、大白话讲通信协议—详解主流通讯协议4.1、文本协议4.2、二进制协议4.3、xml协议4.4、可以落地使用的协议 5、私有协议编解码—设计篇…

ArcGIS栅格重采样与算法选择

本文介绍在ArcMap软件中&#xff0c;实现栅格图像重采样的具体操作&#xff0c;以及不同重采样方法的选择依据。 首先&#xff0c;如下图所示&#xff0c;是我们待重采样的栅格图像的属性界面。其中&#xff0c;可以看到此时栅格像元的边长为0.4867左右&#xff08;由于图层是地…

“技术开发最应该做什么?”,聊聊我在服务端开发5年的理解和收获

我们新推出大淘宝技术年度特刊《长期主义&#xff0c;往往从一些小事开始——工程师成长总结专题》&#xff0c;专题收录多位工程师真诚的心路历程与经验思考&#xff0c;覆盖终端、服务端、数据算法、技术质量等7大技术领域&#xff0c;欢迎一起沟通交流。 本文为此系列第二篇…

我的.net视频课程

https://edu.csdn.net/lecturer/222?spm1002.2001.3001.4144

9.并发基础与CAS基本原理

线程的状态/线程的生命周期 初始化 Thread thread new Thread();运行 thread.start(); 运行中状态 ——> 调用yeild进入就绪状态就绪状态 ——> 通过系统调度(获取到cpu时间片),又会进入到运行中状态 等待 调用如下方法就从运行进入到等待状态: Object.wait()、Object.…

2360. 图中的最长环

方法一&#xff1a;拓扑排序加搜索 class Solution { public:bool vis[100005];vector<int>v[100005];int dfs(vector<int> &dist,int st,int step){vis[st]true; // cout<<st<<endl;int res0;for(int i0;i<v[st].size();i){int xv[st][i];if(…

Angular开发之——Angular项目介绍(03)

一 概述 Angular项目目录介绍Angular程序如何启动 二 Angular项目目录介绍 2.1 项目目录 2.2 目录结构说明 工作区配置文件 node_modules &#xff1a;第三方依赖包存放的目录src&#xff1a;应用源代码目录angular-cli.json&#xff1a; Angular命令行工具的配置文件&…

设计模式之桥接模式释义与举例剖析

文章目录 一、前言二、模式定义三、模式结构四、案例具体实现五、 模式优缺点六、 模式使用场景七、 模式总结 一、前言 开始学Java讲继承的时候&#xff0c;总喜欢用一个例子来讲解&#xff0c;那就是画图形。这里有一个画笔&#xff0c;可以画正方形、长方形、圆形。除了画出…

OpenCV-答题卡识别-四点透视变换

目录 答题卡识别图片读取四点透视变换 划出区域处理选择题区域处理准考证号区域处理科目区域得分导出结果 封装成品 答题卡识别 使用opencv技术&#xff0c;实现对答题卡的自动识别&#xff0c;并进行答题结果的统计 技术目的&#xff1a; 能够捕获答题卡中的每个填涂选项;将…

【新星计划-2023】TCP三次握手和四次挥手讲解

关于TCP三次握手和四次挥手&#xff0c;各位想必在读大学的时候或者是在面试的时候一定遇到过&#xff0c;三次握手和四次挥手本身是不是太难的&#xff0c;但它容易忘&#x1f61e;&#xff0c;今天我就在这里给大家讲解一下三次握手与四次挥手。 一、三次挥手 TCP三次握手建…

我,大厂P9,找不到工作

作者| 老W 编辑| Emma 来源| 技术领导力(ID&#xff1a;jishulingdaoli) K哥写在前面的话&#xff1a;这是一位读者投稿&#xff0c;读者老W讲述了自己从大厂P9失业后、再就业的故事&#xff0c;并总结了自己的心路历程&#xff0c;很真实的记录与思考&#xff0c;值得大家借…

SensorX2car:在道路场景下的完成传感器到车体坐标系标定

文章&#xff1a;SensorX2car: Sensors-to-car calibration for autonomous driving in road scenarios 作者&#xff1a;Guohang Yan, Zhaotong Luo, Zhuochun Liu and Yikang Li 编辑&#xff1a;点云PCL 代码&#xff1a;https://github.com/OpenCalib/SensorX2car 作者单位…

IDEA+maven+Springboot工程创建超详细过程示例

IDEAmavenSpringboot工程创建超详细过程示例 1、IDEA、Maven下载安装及IDEA配置Maven教程2、IDEAmavenSpringboot的web工程创建示例2.1 SpringBoot简介2.2 maven形式创建sprintboot工程2.3 导入相关依赖2.4 创建SpringBoot启动类2.5 创建 Controller2.6 启动项目2.7 配置端口信…

springboot第22集:security,Lombok,token,redis

Spring Security是一个基于Spring框架的权限管理框架&#xff0c;用于帮助应用程序实现身份验证和授权功能。它可以为Web应用程序、REST API和方法级安全性提供支持&#xff0c;并支持各种认证方式。 Spring Security最初是Acegi Security的前身&#xff0c;但由于其配置繁琐而…

【第六期】Apache DolphinScheduler 每周 FAQ 集锦

点击蓝字 关注我们 摘要 为了让 Apache DolphinScheduler 的广大用户和爱好者对于此项目的疑问得到及时快速的解答&#xff0c;社区特发起此次【每周 FAQ】栏目&#xff0c;希望可以解决大家的实际问题。 关于本栏目的要点&#xff1a; 本栏目每周将通过腾讯文档&#xff08;每…

用排列组合来编码通信(六)——魔术《5张牌的预言》的魔术拓展之《My Fitch Four Glory》...

早点关注我&#xff0c;精彩不错过&#xff01; 在上一篇中&#xff0c;我们介绍了《5张牌的预言》这个魔术的一个精彩的扩展表演《Eigens Value》&#xff0c;把这个魔术和数学性质的结合做到了极致&#xff0c;相关内容请戳&#xff1a; 用排列组合来编码通信&#xff08;五&…

第二十二章 Unity 光照贴图

光照贴图过程将预先计算场景中静态物体表面的亮度&#xff0c;并将结果存储在称为“光照贴图”的纹理中供以后使用。光照贴图可以包含直接光照和间接光照&#xff0c;以及阴影效果。但是&#xff0c;烘焙到光照贴图中的数据无法在运行时更改&#xff0c;这就是为什么移动静态物…