二叉树的层序遍历经典问题(算法村第六关白银挑战)

基本的层序遍历与变换

二叉树的层序遍历

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

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

示例 1:

img

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

示例 2:

输入:root = [1]
输出:[[1]]

示例 3:

输入:root = []
输出:[]
public static List<List<Integer>> levelOrder(TreeNode root)
{//特判,否则queue.offer(root)会抛出NullPointerExceptionif (root == null)return new ArrayList<List<Integer>>();ArrayList<List<Integer>> ans = new ArrayList<>();ArrayDeque<TreeNode> queue = new ArrayDeque<>();queue.offer(root);while (!queue.isEmpty()){//获取当前层的结点个数int size = queue.size();//该层结点的值ArrayList<Integer> list = new ArrayList<>();//遍历当前层for (int i = 0; i< size; i++){TreeNode t = queue.remove();list.add(t.val);if (t.left != null)queue.offer(t.left);if (t.right != null)queue.offer(t.right);}//将这一层结点的值加入答案ans.add(list);}return ans;
}

自底而上的层序遍历

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

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

示例 1:

img

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

在遍历完一层节点之后,将存储该层节点值的列表 list 添加到结果列表 ans 的头部即可。

为了时间复杂度,ans 使用链式结构的结构。在 ans 头部添加一层节点值的列表 list 的时间复杂度是 O(1)

public List<List<Integer>> levelOrderBottom(TreeNode root)
{//特判,否则queue.offer(root)会抛出NullPointerExceptionif (root == null)return new ArrayList<List<Integer>>();ArrayList<List<Integer>> ans = new ArrayList<>();ArrayDeque<TreeNode> queue = new ArrayDeque<>();queue.offer(root);while (!queue.isEmpty()){//获取当前层的结点个数int size = queue.size();//该层结点的值ArrayList<Integer> list = new ArrayList<>();//遍历当前层for (int i = 0; i< size; i++){TreeNode t = queue.remove();list.add(t.val);if (t.left != null)queue.offer(t.left);if (t.right != null)queue.offer(t.right);}//将这一层结点的值插入答案的头部ans.add(0,list);}return ans;
}

锯齿形层序遍历

103. 二叉树的锯齿形层序遍历 - 力扣(LeetCode)

给你二叉树的根节点 root ,返回其节点值的 锯齿形层序遍历 。(即先从左往右,再从右往左进行下一层遍历,以此类推,层与层之间交替进行)。

示例 1:

img

输入:root = [3,9,20,null,null,15,7]
输出:[[3],[20,9],[15,7]]
用双端队列维护当前层节点的存储

双端队列可以队头或队尾插入元素。

层序遍历顺序不变,但对当前层节点的存储,我们维护一个变量 isOrderLeft ,记录结点存储是从左至右还是从右至左

  1. 若从左至右,则采用头插法。该层第一个元素在此层遍历结束后,会出现在list的末端
  2. 若从右至左,则采用尾插法。该层第一个元素在此层遍历结束后,会出现在list的首端

最后需要注意的是,往 ans 添加 list 时,需要转换一下 list 的类型

public List<List<Integer>> zigzagLevelOrder(TreeNode root){//特判,否则queue.offer(root)会抛出NullPointerExceptionif (root == null)return new ArrayList<List<Integer>>();ArrayList<List<Integer>> ans = new ArrayList<>();ArrayDeque<TreeNode> queue = new ArrayDeque<>();queue.offer(root);boolean isOrderLeft = true;while (!queue.isEmpty()){//获取当前层的结点个数int size = queue.size();//该层结点的值。用一个双端队列存储ArrayDeque<Integer> deque = new ArrayDeque<>();//遍历当前层for (int i = 0; i< size; i++){TreeNode t = queue.remove();if (isOrderLeft)deque.offerLast(t.val);elsedeque.offerFirst(t.val);if (t.left != null)queue.offer(t.left);if (t.right != null)queue.offer(t.right);}//这一层结点的值经过转换后加入答案ans.add(new LinkedList<>(deque));isOrderLeft = !isOrderLeft; //交替进行}return ans;}

N叉树的层序遍历

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

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

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

示例 1:

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

示例 2:

img
输入:root = [1,null,2,3,4,5,null,null,6,7,null,8,null,9,10,null,null,11,null,12,null,13,null,null,14]
输出:[[1],[2,3,4,5],[6,7,8,9,10],[11,12,13],[14]]
结点类型
public class Node
{public int val;public List<Node> children;public Node() {}public Node(int _val) {val = _val;}public Node(int _val, List<Node> _children) {val = _val;children = _children;}
}
public List<List<Integer>> levelOrder(Node root)
{//特判,否则queue.offer(root)会抛出NullPointerExceptionif (root == null)return new ArrayList<List<Integer>>();ArrayList<List<Integer>> ans = new ArrayList<>();ArrayDeque<Node> queue = new ArrayDeque<>();queue.offer(root);while (!queue.isEmpty()){//获取当前层的结点个数int size = queue.size();//该层结点的值ArrayList<Integer> list = new ArrayList<>();//遍历当前层for (int i = 0; i< size; i++){Node t = queue.remove();list.add(t.val);//将当前结点的所有孩子加入队列for (Node child : t.children){queue.offer(child);}}//将这一层结点的值加入答案ans.add(list);}return ans;
}

处理每层元素

在每个树行中找最大值

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

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

示例1:

img
输入: root = [1,3,2,5,3,null,9]
输出: [1,3,9]
维护每层的最大值即可
public List<Integer> largestValues(TreeNode root)
{//特判,否则queue.offer(root)会抛出NullPointerExceptionif (root == null)return new ArrayList<>();//存储每层结点的最大值ArrayList<Integer> list = new ArrayList<>();ArrayDeque<TreeNode> queue = new ArrayDeque<>();queue.offer(root);while (!queue.isEmpty()){//获取当前层的结点个数int size = queue.size();//当前层结点的最大值int maxOfLevel = Integer.MIN_VALUE;//遍历当前层for (int i = 0; i< size; i++){TreeNode t = queue.remove();maxOfLevel = Math.max(maxOfLevel, t.val);if (t.left != null)queue.offer(t.left);if (t.right != null)queue.offer(t.right);}list.add(maxOfLevel);}return list;
}

二叉树的层平均值

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

给定一个非空二叉树的根节点 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]
public List<Double> averageOfLevels(TreeNode root)
{//特判,否则queue.offer(root)会抛出NullPointerExceptionif (root == null)return new ArrayList<>();//存储每层结点的最大值ArrayList<Double> list = new ArrayList<>();ArrayDeque<TreeNode> queue = new ArrayDeque<>();queue.offer(root);while (!queue.isEmpty()){//获取当前层的结点个数int size = queue.size();//当前层结点值的和double sum = 0;//遍历当前层for (int i = 0; i< size; i++){TreeNode t = queue.remove();sum += t.val;if (t.left != null)queue.offer(t.left);if (t.right != null)queue.offer(t.right);}//计算平均值并添加到列表list.add(sum / size);}return list;
}

二叉树的右视图

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

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

示例 1:

img

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

层序遍历,记录每层最后一个元素即可

public List<Integer> rightSideView(TreeNode root)
{//特判,否则queue.offer(root)会抛出NullPointerExceptionif (root == null)return new ArrayList<>();//存储每层的最后一个结点ArrayList<Integer> list = new ArrayList<>();ArrayDeque<TreeNode> queue = new ArrayDeque<>();queue.offer(root);while (!queue.isEmpty()){//获取当前层的结点个数int size = queue.size();//遍历当前层for (int i = 0; i < size; i++){TreeNode t = queue.remove();//记录当前层最后一个元素if (i == size - 1)list.add(t.val);if (t.left != null)queue.offer(t.left);if (t.right != null)queue.offer(t.right);}}return list;
}

最底层最左边的结点

513. 找树左下角的值 - 力扣(LeetCode)

给定一个二叉树的 根节点 root,请找出该二叉树的 最底层 最左边 节点的值。

假设二叉树中至少有一个节点。

示例 1:

img

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

示例 2:

img
输入: [1,2,3,4,null,5,6,null,null,7]
输出: 7
从右往左层序遍历

从右向左层次遍历, 最后一个访问的节点必然是最底层最左侧叶子节点。只需调整一下左右孩子加入队列的次序即可

public int findBottomLeftValue(TreeNode root)
{ArrayDeque<TreeNode> queue = new ArrayDeque<>();queue.offer(root);TreeNode t = null;while (!queue.isEmpty()){//获取当前层的结点个数int size = queue.size();//遍历当前层for (int i = 0; i < size; i++){t = queue.remove();//右孩子先于左孩子放入队列if (t.right != null)queue.offer(t.right);if (t.left != null)queue.offer(t.left);}}return t.val;   //返回最底层最右边的结点的值
}

左叶子之和

404. 左叶子之和 - 力扣(LeetCode)

给定二叉树的根节点 root ,返回所有左叶子之和。

示例 1:

img

输入: root = [3,9,20,null,null,15,7] 
输出: 24 
解释: 在这个二叉树中,有两个左叶子,分别是 915,所以返回 24

提示:

  • 节点数在 [1, 1000] 范围内

广度优先遍历

在BFS的过程中添加一个判断是否为左叶子结点的条件语句即可

public int sumOfLeftLeaves(TreeNode root)
{int answer = 0;ArrayDeque<TreeNode> queue = new ArrayDeque<>();queue.offer(root);while (!queue.isEmpty()){//获取当前层的结点个数int size = queue.size();//遍历当前层for (int i = 0; i < size; i++){TreeNode t = queue.remove();if (t.left != null){//如果t的左孩子是叶子结点if(t.left.left == null && t.left.right == null)answer += t.left.val;elsequeue.offer(t.left);}if (t.right != null)queue.offer(t.right);}}return answer;
}

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【合阳新起点公益】“关爱留守儿童 守护牙齿健康”牙膏发放活动

为了关爱儿童的口腔健康&#xff0c;帮助他们改善生活状况&#xff0c;养成良好的刷牙习惯。合阳县未成年人保护中心、合阳县新起点公益服务中心组织链接到汕头市惠泽人志愿服务中心&#xff0c;为孩子们申请到一批儿童爱心牙膏套盒&#xff0c;分别于2023年12月22日、12月30日…

第一课:Transformer

第一课&#xff1a;Transformer 文章目录 第一课&#xff1a;Transformer1、学习总结&#xff1a;什么是语言模型&#xff1f;大语言模型&#xff08;LLM&#xff09;技术演变史注意力机制Transformer结构课程ppt及代码地址 2、学习心得&#xff1a;3、经验分享&#xff1a;4、…

uniappVue3版本中组件生命周期和页面生命周期的详细介绍

一、什么是生命周期&#xff1f; 生命周期有多重叫法&#xff0c;有叫生命周期函数的&#xff0c;也有叫生命周期钩子的&#xff0c;还有钩子函数的&#xff0c;其实都是代表&#xff0c;在 Vue 实例创建、更新和销毁的不同阶段触发的一组钩子函数&#xff0c;这些生命周期函数…

面试题:你如何理解 System.out.println()?

文章目录 前言首先分析System源码&#xff1a;out源码分析println分析 前言 如果你能自己读懂System.out.println()&#xff0c;就真正了解了Java面向对象编程的含义。 面向对象编程即创建了对象&#xff0c;所有的事情让对象帮亲力亲为&#xff08;即对象调用方法&#xff09…

2023年广东省网络安全A模块(笔记详解)

模块A 基础设施设置与安全加固 一、项目和任务描述&#xff1a; 假定你是某企业的网络安全工程师&#xff0c;对于企业的服务器系统&#xff0c;根据任务要求确保各服务正常运行&#xff0c;并通过综合运用登录和密码策略、流量完整性保护策略、事件监控策略、防火墙策略等多…

Windows通过注册表修改socket缓冲区大小的方法

在 Windows 通过修改注册表来更改 UDP 缓冲区的大小&#xff0c;按照以下步骤进行操作&#xff1a; 打开注册表编辑器&#xff1a;按下 Win R 键&#xff0c;然后输入 "regedit" 并点击 "确定"。 导航到以下路径&#xff1a;HKEY_LOCAL_MACHINE\System\C…

线程同步之:QMutex\QMutexLocker

1、基于互斥量的线程同步类QMutex 2、lock() 与 unlock()必须配对使用。 2.1 lock() unlock() 2.2 tryLock() unlock() 3、QMutexLocker()是另一个简化了互斥量处理的类。在QMutexLocker实例变量的“生命周期”内的代码段 得到保护。 QMutexLocker的构造函数接受要给互斥量…

12.递归汉诺塔

使用递归实现汉诺塔 public class Main {public static void move(char pos1,char pos2) {System.out.print(pos1" > "pos2" ");}public static void han(int n,char pos1,char pos2,char pos3) {if(n 1) {move(pos1,pos3);return ;}han(n-1,pos1,pos…

项目初始化脚手架搭建

项目初始化脚手架搭建 仓库地址 easy-web: 一个快速初始化SpringBoot项目的脚手架 (gitee.com) 目前这个项目还是个单体项目&#xff0c;后续笔者有时间可能会改造成父子工程项目&#xff0c;将通用模块抽象出来&#xff0c;有兴趣的小伙伴也可以自行 CV 改造。 1、项目初始化…

接口测试测什么?一个简单问题把我难住了!

那么设计测试用例时我们主要考虑如下几个方面&#xff1a; 01、功能测试 接口的功能是否正确实现了 接口是否按照设计文档中来实现 &#xff08;比如username参数写为了user&#xff0c;那么这就不符合&#xff0c;因为接口文档在整个开发中都需要使用&#xff0c;所以接口实…

电商平台低价品牌要如何处理

低价会影响品牌渠道的长期发展&#xff0c;同时还会衍生很多问题&#xff0c;如为了追求低价而导致的店铺窜货、商品假货等&#xff0c;对于渠道来说&#xff0c;都是要及时解决的问题&#xff0c;否则渠道乱了&#xff0c;最终腐蚀的是品牌价值&#xff0c;同时还会影响经销商…

键盘事件,DOM事件流,e.keyup, e.keypress,BOM里的window对象, setInterval, clearInterval

目录 事件委托&#xff08;事件代理&#xff09; 键盘事件 ​编辑e.keycode 京东快递单号查询 页面加载事件 定时器setTimeout 清除定时器 监听事件&#xff1a; 传统的onclick对于后面的注册事件会覆盖前面的注册事件 方法监听注册方式&#xff1a;addEventListener() …