代码随想录算法训练营day15||二叉树part02、102.二叉树的层序遍历、 226.翻转二叉树(优先掌握递归)、101. 对称二叉树 (优先掌握递归)

102.二叉树的层序遍历

题目:给你一个二叉树,请你返回其按 层序遍历 得到的节点值。 (即逐层地,从左到右访问所有节点)。

接下来我们再来介绍二叉树的另一种遍历方式:层序遍历。

层序遍历一个二叉树。就是从左到右一层一层的去遍历二叉树。这种遍历的方式和我们之前讲过的都不太一样。

需要借用一个辅助数据结构即队列来实现,队列先进先出,符合一层一层遍历的逻辑,而用栈先进后出适合模拟深度优先遍历也就是递归的逻辑。

而这种层序遍历方式就是图论中的广度优先遍历,只不过我们应用在二叉树上。

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

class Solution {//定义了一个名为resList 的成员变量,类型为List<List<Integer>>,用于存储层序遍历结果。//这个成员变量是一个二维列表,每个元素是一个!列表!,表示二叉树的一层节点值。public List<List<Integer>> resList = new ArrayList<List<Integer>>();//定义了一个名为levelOrder的公共方法,该方法接受一个TreeNode类型的参数root,表示二叉树的根节点。//该方法的返回类型是List<List<Integer>>,表示二叉树的层序遍历结果。public List<List<Integer>> levelOrder(TreeNode root) {checkFun02(root);//调用checkFun02 方法进行层序遍历。return resList;}//定义了一个名为checkFun02 的方法,用于实现二叉树的层序遍历。//首先检查当前节点node 是否为空,如果为空,则直接返回,表示无需进行层序遍历。public void checkFun02(TreeNode root){if(root == null) return;//创建一个名为que的队列,用于存储待访问的节点,并将根节点root入队列。Queue<TreeNode> que = new LinkedList<TreeNode>();que.offer(root);//offer():将元素添加到队尾,如果成功,则返回true。//使用循环来遍历队列中的节点,直到队列为空为止。while(!que.isEmpty()){//在内层循环中,首先创建一个名为itemList的列表,用于存储当前层的节点值,//并获取当前队列的大小,表示当前层的节点个数。List<Integer> itemList = new ArrayList<Integer>();int len = que.size();//这里一定要使用固定大小size,不要使用que.size(),因为que.size是不断变化的while(len > 0){//从队列中弹出一个节点tmpNode,并将其值添加到当前层的列表itemList中。TreeNode tmpNode = que.poll();//poll():将队首的元素删除,并返回该元素。itemList.add(tmpNode.val); if(tmpNode.left != null)  que.offer(tmpNode.left);if(tmpNode.right != null)  que.offer(tmpNode.right);len--;}//当前层的所有节点都处理完毕后,将当前层的列表itemList添加到结果列表resList中,并进入下一层的处理。resList.add(itemList);}}//至此,二叉树的层序遍历完成,结果存储在成员变量resList中,返回给调用者。
}

107.二叉树的层次遍历 II

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

 思路:相对于102.二叉树的层序遍历,就是最后把result数组反转一下就可以了。

class Solution {public List<List<Integer>> resList = new ArrayList<List<Integer>>();public List<List<Integer>> levelOrderBottom(TreeNode root) {//checkFun01(root,0);checkFun02(root);//创建一个新的二维列表 result,用于存储按自底向上顺序排列的层序遍历结果。List<List<Integer>> result = new ArrayList<>();//使用循环遍历原始的层序遍历结果 resList,从最后一层开始,逐层向上遍历。//在遍历过程中,将每一层的节点列表添加到 result 中。for (int i = resList.size() - 1; i >= 0; i-- ) {result.add(resList.get(i));}return result;//返回按自底向上顺序排列的层序遍历结果列表 result。}public void checkFun02(TreeNode node) {if (node == null) return;Queue<TreeNode> que = new LinkedList<TreeNode>();que.offer(node);while (!que.isEmpty()) {List<Integer> itemList = new ArrayList<Integer>();int len = que.size();while (len > 0) {TreeNode tmpNode = que.poll();itemList.add(tmpNode.val);if (tmpNode.left != null) que.offer(tmpNode.left);if (tmpNode.right != null) que.offer(tmpNode.right);len--;}resList.add(itemList);}}
}

  在102基础上稍作改动:

     //创建一个新的二维列表 result,用于存储按自底向上顺序排列的层序遍历结果。

        List<List<Integer>> result = new ArrayList<>();

     //使用循环遍历原始的层序遍历结果 resList,从最后一层开始,逐层向上遍历。

     //在遍历过程中,将每一层的节点列表添加到 result 中。

        for (int i = resList.size() - 1; i >= 0; i-- ) {

            result.add(resList.get(i));

        }

        return result; //返回按自底向上顺序排列的层序遍历结果列表 result。

 199.二叉树的右视图

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

思路:层序遍历的时候,判断是否遍历到单层的最后面的元素,如果是,就放进result数组中,随后返回result就可以了。

class Solution {/*** 解法:队列,迭代。* 每次返回每层的最后一个字段即可。* 小优化:每层右孩子先入队。代码略。*/public List<Integer> rightSideView(TreeNode root) {
//定义了一个名为rightSideView的公共方法,该方法接受一个TreeNode类型的参数root,表示二叉树的根节点。
//该方法的返回类型是List<Integer>,表示二叉树每层最右侧节点的值。//创建一个名为 resList 的列表,用于存储每层最右侧节点的值。List<Integer> resList = new ArrayList<>();//创建一个名为 que 的双端队列,用于进行层序遍历。//这里选择双端队列的原因是为了让每层的右孩子先入队,以便后续直接取出最后一个节点。Deque<TreeNode> que = new LinkedList<>();//如果根节点为空,则直接返回空列表。if (root == null) {return resList;}//将根节点入队。que.offerLast(root);//使用循环遍历队列中的节点,直到队列为空为止。while (!que.isEmpty()) {//获取当前层的节点数levelSize,并使用内层循环遍历当前层的所有节点。int levelSize = que.size();//从队列中弹出一个节点tmpNode,并将其左右孩子节点依次入队。for (int i = 0; i < levelSize; i++) {TreeNode tmpNode = que.pollFirst();if (tmpNode.left != null) {que.addLast(tmpNode.left);}if (tmpNode.right != null) {que.addLast(tmpNode.right);}//如果当前节点是当前层的最后一个节点,则将其值添加到结果列表resList中。if (i == levelSize - 1) {resList.add(tmpNode.val);} //继续处理下一层的节点。}}return resList;//返回存储了每层最右侧节点值的列表resList。}
}

637.二叉树的层平均值

给定一个非空二叉树, 返回一个由每层节点平均值组成的数组。

思路:本题就是层序遍历的时候把一层求个总和在取一个均值。

class Solution {public List<Double> averageOfLevels(TreeNode root) {//定义了一个名为averageOfLevels 的公共方法,该方法接受一个TreeNode类型的参数root,表示二叉树的根节点。//该方法的返回类型是List<Double>,表示二叉树每层节点值的平均值。//创建一个名为resList的列表,用于存储每层节点值的平均值。List<Double> resList = new ArrayList<>();//创建一个名为 que 的双端队列,用于进行层序遍历。Deque<TreeNode> que = new LinkedList<>();if(root == null){return resList;}//将根节点入队。que.offerLast(root);//使用循环遍历队列中的节点,直到队列为空为止。while(!que.isEmpty()){//获取当前层的节点数levelSize,并使用内层循环遍历当前层的所有节点。int levelSize = que.size();double levelSum = 0.0;for(int i = 0; i < levelSize; i++){//从队列中弹出一个节点poll,并将其值加到当前层的和levelSum 中。TreeNode tmpNode = que.pollFirst();levelSum += tmpNode.val;//然后将其左右孩子节点依次入队。if(tmpNode.left != null) que.addLast(tmpNode.left);if(tmpNode.right != null) que.addLast(tmpNode.right);}resList.add(levelSum / levelSize);//计算当前层节点值的平均值,并将其添加到结果列表resList中。}return resList;//返回存储了每层节点值平均值的列表list。}
}

429.N叉树的层序遍历

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

思路:这道题依旧是模板题,只不过一个节点有多个孩子了

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

您需要在二叉树的每一行中找到最大的值。

思路:层序遍历,取每一层的最大值

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

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

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

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

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

思路:本题依然是层序遍历,只不过在单层遍历的时候记录一下本层的头部节点,然后在遍历的时候让前一个节点指向本节点就可以了

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

思路:这道题目说是二叉树,但116题目说是完整二叉树,其实没有任何差别,一样的代码一样的逻辑一样的味道

104.二叉树的最大深度

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

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

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

示例:给定二叉树 [3,9,20,null,null,15,7],

 111.二叉树的最小深度

思路:相对于 104.二叉树的最大深度 ,本题还也可以使用层序遍历的方式来解决,思路是一样的。需要注意的是,只有当左右孩子都为空的时候,才说明遍历的最低点了。如果其中一个孩子为空则不是最低点

总结:

二叉树的层序遍历,就是图论中的广度优先搜索在二叉树中的应用,需要借助队列来实现(此时又发现队列的一个应用了)。

来吧,一口气打十个:

  • 102.二叉树的层序遍历(opens new window)
  • 107.二叉树的层次遍历II(opens new window)
  • 199.二叉树的右视图(opens new window)
  • 637.二叉树的层平均值(opens new window)
  • 429.N叉树的层序遍历(opens new window)
  • 515.在每个树行中找最大值(opens new window)
  • 116.填充每个节点的下一个右侧节点指针(opens new window)
  • 117.填充每个节点的下一个右侧节点指针II(opens new window)
  • 104.二叉树的最大深度(opens new window)
  • 111.二叉树的最小深度

 226.翻转二叉树

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

题外话:这道题目是非常经典的题目,也是比较简单的题目(至少一看就会)。

但正是因为这道题太简单,一看就会,一些同学都没有抓住起本质,稀里糊涂的就把这道题目过了。 如果做过这道题的同学也建议认真看完,相信一定有所收获!

注意只要把每一个节点的左右孩子翻转一下,就可以达到整体翻转的效果

这道题目使用前序遍历和后序遍历都可以,唯独中序遍历不方便,因为中序遍历会把某些节点的左右孩子翻转了两次!建议拿纸画一画,就理解了

那么层序遍历可以不可以呢?依然可以的!只要把每一个节点的左右孩子翻转一下的遍历方式都是可以的!

递归法:

class Solution {//递归法
/** 前后序遍历都可以(这里代码给出的后序遍历,先递归处理左子树,然后递归处理右子树,最后再处理根节点。)* 中序不行,因为先左孩子交换孩子,再根交换孩子(做完后,右孩子已经变成了原来的左孩子),再右孩子交换孩子(此时其实是对原来的左孩子做交换)
*/public TreeNode invertTree(TreeNode root) {if (root == null) {//递归终止条件return root;}//递归地对当前节点的左右子树进行反转,即先对左子树进行反转,再对右子树进行反转。invertTree(root.left);invertTree(root.right);swapChildren(root); //调用swapChildren 方法交换当前节点的左右子节点。return root;//返回反转后的二叉树的根节点。} //定义了一个私有方法swapChildren,用于交换节点的左右子节点。//这个方法接受一个节点root,将其左右子节点进行交换private void swapChildren(TreeNode root){TreeNode tmp = root.left;root.left = root.right;root.right = tmp;}
}

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

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

相关文章

Centos7之忘记Root用户密码的处理方式

Centos7之忘记Root用户密码的处理方式 文章目录 Centos7之忘记Root用户密码的处理方式1.场景描述2. 重置密码1. 重启系统进入编辑界面2. 按方向键下键↓&#xff0c;找到设置语言的地方3. 进入bash界面后&#xff0c;可以输入passwd命令重新设置root密码 1.场景描述 长时间未使…

前后端分离nodejs+vue流浪狗宠物领养公益网站

1.发现公益&#xff1a;主要是根据社会上的调研&#xff0c;来收集的社会上有关流浪狗的公益活动&#xff0c;发布在公益网站上能被更多人发现&#xff0c;主要让更多人能参与到公益活动中来&#xff0c;并调动群众的同情心和爱心&#xff0c;借此希望在养宠物的主人能避免自己…

vue3 之 商城项目—项目搭建起步

1.创建项目 1️⃣ npm init vuelatest2️⃣ npm install3️⃣ npm run dev4️⃣目录调整 2.git管理项目 基于creact-vue创建出来的项目默认没有初始化git仓库&#xff0c;需要我们手动初始化 执行命令 git init git add. git commit -m init3.项目起步—配置别名路径联…

前端开发_AJAX基本使用

AJAX概念 AJAX是异步的JavaScript和XML(Asynchronous JavaScript And XML)。 简单点说&#xff0c;就是使用XMLHttpRequest对象与服务器通信。 它可以使用JSON&#xff0c;XML&#xff0c;HTML和text文本等格式发送和接收数据。 AJAX最吸引人的就是它的“异步"特性&am…

顶级思维方式——对优秀人才的定义

目录 1、乔布斯对优秀人才的定义 2、 乔布斯对优秀人才的管理 3、感到压力焦虑的时候怎么办 注&#xff1a; 以下内容均来自乔布斯、贝索斯的采访视频摘录 1、乔布斯对优秀人才的定义 乔布斯&#xff1a; 公司规模变大之后&#xff0c;就会变得循规蹈矩。他们觉得只要遵守流…

MySQL数据库-索引概念及其数据结构、覆盖索引与回表查询关联、超大分页解决思路

索引是帮助mysql高效获取数据的数据结构,主要用来提高检索的效率,降低数据库的IO成本(输入输出成本&#xff08;Input-Output Cost&#xff09;),同时通过索引对数据进行排序也能降低数据排序的成本,降低了CPU的消耗。 Mysql的默认存储引擎InnoDB&#xff0c;InnoDB采用的B树的…

肯尼斯·里科《C和指针》第12章 使用结构和指针(2)双链表

12.3 双链表 单链表的替代方案就是双链表。在一个双链表中&#xff0c;每个节点都包含两个指针——指向前一个节点的指针和指向后一个节点的指针。这可以使我们以任何方向遍历双链表&#xff0c;甚至可以随意在双链表中访问。下面的图展示了一个双链表。 下面是节点类型的声明&…

算法刷题:移动零

移动零 .题目链接详解curdesc算法原理 答案 . 题目链接 移动零 详解 题目要求我们要把数组中所有的零都移动到数组的末尾,且要求其余数字顺序不改变.这道题,我们使用到的是双指针算法: 利用两个指针,将数组分为三个部分, 三个区间分别为 [0,desc][desc1,cur-1][cur,n-1] 在…

算法学习——LeetCode力扣二叉树篇1

算法学习——LeetCode力扣二叉树篇1 144. 二叉树的前序遍历 144. 二叉树的前序遍历 - 力扣&#xff08;LeetCode&#xff09; 描述 给你二叉树的根节点 root &#xff0c;返回它节点值的 前序 遍历。 示例 示例 1&#xff1a; 输入&#xff1a;root [1,null,2,3] 输出&a…

电子电器架构 —— 区域控制器是未来架构的正解吗?

电子电器架构 —— 区域控制器是未来架构的正解吗? 我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师(Wechat:gongkenan2013)。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 本就是小人物,输了就是输了,不要在意别人怎么看自己。江湖一碗茶…

C++ //练习 5.5 写一段自己的程序,使用if else语句实现把数字成绩转换成字母成绩的要求。

C Primer&#xff08;第5版&#xff09; 练习 5.5 练习 5.5 写一段自己的程序&#xff0c;使用if else语句实现把数字成绩转换成字母成绩的要求。 环境&#xff1a;Linux Ubuntu&#xff08;云服务器&#xff09; 工具&#xff1a;vim 代码块 /***************************…

视觉开发板—K210自学笔记(四)

在点灯之后&#xff0c;我们就需要饿补一下相关的编程基础知识&#xff0c;了解基本语法&#xff0c;加深底蕴才能编写出更好的程序来。由于 MaixPy 是基于 MicroPython 之上进行开发构建的&#xff0c;提供给用户最终的接口是 Micropython &#xff0c;所以在使用 MaixPy 开发…

C语言-----自定义类型-----结构体枚举联合

结构体和数组一样&#xff0c;都是一群数据的集合&#xff0c;不同的是数组当中的数据是相同的类型&#xff0c;但是结构体中的数据类型可以不相同&#xff0c;结构体里的成员叫做成员变量 结构体类型是C语言里面的一种自定义类型&#xff0c;我们前面已经了解到过int,char,fl…

选择影视行业创业的原因,影视从业者创业成功的秘密

一、教程描述 本套教程是面向影视从业者的创业教程&#xff0c;主讲人将把自己的创业经验、行业观察、成长心得分享给大家。如果你正在创业&#xff0c;这门课可以让你飞速成长、弯道超车。主讲人积累的行业经验&#xff0c;会让你比大多数同行站的更高&#xff0c;看的更宽。…

KEIL-MDK的时间戳之time.h 结合gd32f1的RTC应用

KEIL-MDK的时间戳之time.h 的应用 1 时间戳介绍 现在物联网产品的在进行通讯的时候&#xff0c;需要加入时间戳的这个信息参数&#xff0c;方便服务器和产品之间交换时间信息。 时间戳是计算机系统中用来表示日期和时间的一种方式&#xff0c;通常是一个数字或者一串字符&am…

【DDD】学习笔记-统一语言与领域分析模型

无论你采用什么样的软件开发过程&#xff0c;对于一个复杂的软件系统&#xff0c;都必然需要通过分析阶段对问题域展开分析&#xff0c;如此才能有的放矢地针对该软件系统的需求寻找设计上的解决方案。在领域驱动设计中&#xff0c;分析阶段完全围绕着“领域”为中心展开&#…

从信息隐藏到功能隐藏

本文主要记录复旦大学张新鹏教授于2022年12月在第三届CSIG中国媒体取证与安全大会上的汇报

微信小程序 民宿预订租赁系统uniApp

通过山青水磨APP办理租房相关业务&#xff0c;线上解决预定、退订的业务&#xff0c;旅客在使用时更加灵活&#xff0c;实现了快速找房&#xff0c;在线沟通、便捷租赁等操作&#xff0c;除此以外&#xff0c;还能帮助旅客获取周边资讯、当地特色活动服务&#xff0c;提升旅客的…

1-3 mininet中使用python API直接拓扑定义以及启动方式对比

作为SDN网络中搭建拓扑非常重要的仿真平台&#xff0c;我们可以使用mininet默认的库内拓扑文件&#xff0c;也可以使用python语言进行自定义拓扑。使用python进行拓扑定义时&#xff0c;不同的定义方式将导致其启动的方式由所不同。 一、采用最原始的命令启动方式&#xff1a; …

Python 视频转场特效处理笔记

本文参考Python-OpenCV 实现美图秀秀视频剪辑效果【特效】_opencv 多张图片 视频 特效-CSDN博客 最近研究了点python处理视频相关的东西&#xff0c;本文展示特效包括&#xff0c;竖向开幕/横向开幕&#xff0c;渐隐/渐显&#xff0c;推近/拉远&#xff0c;方形开幕&#xff0…