102. 二叉树的层序遍历递归法:深度优先搜索的巧妙应用
二叉树的层序遍历是一种经典的遍历方式,它要求按层级逐层访问二叉树的节点。通常我们会使用队列来实现层序遍历,但递归法也是一种可行且有趣的思路。本文将深入探讨递归法解决二叉树层序遍历的核心难点,并结合代码和模拟过程进行详细讲解。
一、题目描述
给定一个二叉树的根节点 root
,返回其节点值的层序遍历结果,即逐层遍历,从左到右访问所有节点。例如,输入二叉树:
3/ \9 20/ \15 7
输出结果应为:
[[3], [9, 20], [15, 7]]
。
二、核心难点分析
难点1:临时数组的创建逻辑
在递归法中,我们需要预先创建好数组来存储每一层的节点值。这里的关键是如何根据当前节点的深度(层级)来确定将节点值存储到哪个数组中。
- 深度标记:
我们使用一个变量deep
来表示当前节点所在的深度。初始时,根节点的深度为1
。 - 数组创建与填充:
在递归过程中,每当遇到一个新的深度deep
时,我们需要确保res
中已经有足够的子列表来存储该层的节点值。如果res
的大小小于deep
,则创建一个新的空列表并添加到res
中。然后,将当前节点的值添加到res
中对应深度的子列表中。
难点2:递归的具体逻辑实现
递归法的核心在于如何通过递归调用自身来遍历二叉树的每一层。
- 递归终止条件:
当遇到null
节点时,递归结束。这是因为null
节点没有值,也没有子节点,不需要进行任何处理。 - 递归调用:
对于每个非null
节点,我们先将其值添加到对应深度的子列表中,然后分别对其左子节点和右子节点进行递归调用。在递归调用时,深度deep
需要加1
,以表示进入了下一层。
三、解题思路分步解析
步骤1:初始化结果列表和递归调用
public List<List<Integer>> levelOrder(TreeNode root) {List<List<Integer>> res = new ArrayList<>();levelOrderBFS(res, root, 0);return res;
}
这里我们首先创建了一个空的结果列表 res
,然后调用递归函数 levelOrderBFS
开始遍历。初始深度设为 0
,因为在递归函数内部会先将深度加 1
再进行处理。
步骤2:递归函数实现
public void levelOrderBFS(List<List<Integer>> res, TreeNode root, int deep) {if (root == null) {return;}deep++;while (res.size() < deep) {List<Integer> resList = new ArrayList<>();res.add(resList);}res.get(deep - 1).add(root.val);levelOrderBFS(res, root.left, deep);levelOrderBFS(res, root.right, deep);
}
- 深度处理:
首先将深度deep
加1
,因为当前节点的深度比其父节点大1
。 - 数组创建与填充:
使用while
循环检查res
的大小是否小于deep
。如果是,则创建一个新的空列表resList
并添加到res
中。这确保了res
中始终有足够的子列表来存储每一层的节点值。
然后,通过res.get(deep - 1).add(root.val)
将当前节点的值添加到res
中对应深度的子列表中。 - 递归调用:
最后,分别对当前节点的左子节点和右子节点进行递归调用,深度deep
保持不变。这使得递归能够逐层深入,遍历整个二叉树。
四、递归流程模拟
以二叉树 [3, 9, 20, null, null, 15, 7]
为例,结构如下:
3/ \9 20/ \15 7
以下是递归过程的详细模拟:
初始状态
res = []
,deep = 0
,root = 3
第一次递归调用
deep = 1
res.size() = 0 < 1
,创建新列表[]
并添加到res
中,此时res = [[]]
- 将
3
添加到res[0]
中,此时res = [[3]]
- 对
root.left
(即9
)进行递归调用,deep = 1
- 对
root.right
(即20
)进行递归调用,deep = 1
对 9
的递归调用
deep = 2
res.size() = 1 < 2
,创建新列表[]
并添加到res
中,此时res = [[3], []]
- 将
9
添加到res[1]
中,此时res = [[3], [9]]
- 对
root.left
(即null
)进行递归调用,deep = 2
- 对
root.right
(即null
)进行递归调用,deep = 2
对 20
的递归调用
deep = 2
res.size() = 2 == 2
,直接将20
添加到res[1]
中,此时res = [[3], [9, 20]]
- 对
root.left
(即15
)进行递归调用,deep = 2
- 对
root.right
(即7
)进行递归调用,deep = 2
对 15
的递归调用
deep = 3
res.size() = 2 < 3
,创建新列表[]
并添加到res
中,此时res = [[3], [9, 20], []]
- 将
15
添加到res[2]
中,此时res = [[3], [9, 20], [15]]
- 对
root.left
(即null
)进行递归调用,deep = 3
- 对
root.right
(即null
)进行递归调用,deep = 3
对 7
的递归调用
deep = 3
res.size() = 3 == 3
,直接将7
添加到res[2]
中,此时res = [[3], [9, 20], [15, 7]]
- 对
root.left
(即null
)进行递归调用,deep = 3
- 对
root.right
(即null
)进行递归调用,deep = 3
最终结果
res = [[3], [9, 20], [15, 7]]
五、关键知识点总结
1. 递归的基本概念
递归是一种解决问题的方法,它通过将问题分解为更小的子问题,并通过调用自身来解决这些子问题。在二叉树的层序遍历中,我们通过递归调用自身来遍历每一层的节点。
2. 深度优先搜索(DFS)与广度优先搜索(BFS)
递归法本质上是一种深度优先搜索(DFS)的应用。与广度优先搜索(BFS)不同,DFS会沿着一条路径一直深入下去,直到无法继续,然后再回溯到上一层,继续探索其他路径。在层序遍历中,我们通过递归实现了一种特殊的DFS,它能够逐层遍历二叉树的节点。
3. 动态数组的使用
在递归过程中,我们使用了一个动态数组 res
来存储每一层的节点值。动态数组的特点是可以根据需要动态地增加或减少元素,这使得我们能够灵活地处理不同层级的节点数量。
六、完整代码
import java.util.ArrayList;
import java.util.List;class TreeNode {int val;TreeNode left;TreeNode right;TreeNode() {}TreeNode(int val) { this.val = val; }TreeNode(int val, TreeNode left, TreeNode right) {this.val = val;this.left = left;this.right = right;}
}class Solution {public List<List<Integer>> levelOrder(TreeNode root) {List<List<Integer>> res = new ArrayList<>();levelOrderBFS(res, root, 0);return res;}public void levelOrderBFS(List<List<Integer>> res, TreeNode root, int deep) {if (root == null) {return;}deep++;while (res.size() < deep) {List<Integer> resList = new ArrayList<>();res.add(resList);}res.get(deep - 1).add(root.val);levelOrderBFS(res, root.left, deep);levelOrderBFS(res, root.right, deep);}
}
七、总结
递归法解决二叉树的层序遍历问题虽然不像队列法那样直观,但它展示了深度优先搜索在树形结构中的巧妙应用。通过理解临时数组的创建逻辑和递归的具体实现,我们能够更好地掌握递归的思想和技巧。在实际应用中,递归法可能在某些情况下比队列法更简洁高效,尤其是对于一些复杂的树形结构或需要深度优先处理的问题。希望本文的讲解能够帮助读者更好地理解和掌握递归法解决二叉树层序遍历的核心难点。
相关文章:
102. 二叉树的层序遍历递归法:深度优先搜索的巧妙应用
二叉树的层序遍历是一种经典的遍历方式,它要求按层级逐层访问二叉树的节点。通常我们会使用队列来实现层序遍历,但递归法也是一种可行且有趣的思路。本文将深入探讨递归法解决二叉树层序遍历的核心难点,并结合代码和模拟过程进行详细讲解。 …...
Github 2025-05-16 Java开源项目日报 Top9
根据Github Trendings的统计,今日(2025-05-16统计)共有9个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量Java项目9Netty:异步事件驱动的网络应用程序框架 创建周期:5043 天开发语言:Java协议类型:Apache License 2.0Star数量:33219 个Fork数量:…...
MinerU安装(pdf转markdown、json)
在Windows上安装MinerU,参考以下几个文章,可以成功安装,并使用GPU解析。 整体安装教程: MinerU本地化部署教程——一款AI知识库建站的必备工具 MinerU本地化部署可视化界面-CSDN博客 其中安装conda的教程: 一步步教…...
Java卡与SSE技术融合实现企业级安全实时通讯
简介 在数字化转型浪潮中,安全与实时数据传输已成为金融、物联网等高安全性领域的核心需求。本文将深入剖析东信和平的Java卡权限分级控制技术与浪潮云基于SSE的大模型数据推送技术,探索如何将这两项创新技术进行融合,构建企业级安全实时通讯系统。通过从零到一的开发步骤,…...

第31讲 循环缓冲区与命令解析
串口在持续接收数据时容易发生数据黏包(先接收的数据尚未被处理,后面的数据已经将内存覆盖)的情况,循环缓冲区的本质就是将串口接受到的数据马上拷贝到另外一块内存之中。为了避免新来的数据覆盖掉尚未处理的数据,一方…...
mapbox-gl强制请求需要accessToken的问题
vue引入"mapbox-gl": "^2.15.0", 1.13以后得版本,都强制需要验证这个mapboxgl.accessToken。 解决办法:实例化地图的代码中,加入这个: const originalFetch window.fetch; window.fetch function ({ url…...

数据结构(十)——排序
一、选择排序 1.简单选择排序 基本思想:假设排序表为[1,…,n],第i趟排序即从[i,…,n]中选择关键字最小的元素与L[i]交换 eg:给定关键字序列{87,45,78,32,17,65,53&…...

美蛋工具箱:一站式解决图片、视频、音频和文档处理需求的聚合神器
先放下载链接:夸克网盘下载 宝子们,今天不啰嗦,直接给大家安利一款超好用的聚合工具,有需要的小伙伴赶紧码住! 今天要介绍的这款工具叫美蛋工具箱,它是一款聚合类工具。这个软件是绿色版的,聚合了图片工具…...
fastadmin 数据导出,设置excel行高和限制图片大小
fastadmin默认导出图片全部都再一块,而且不在单元格里 话不多说,上代码 修改文件的路径: /public/assets/js/require-table.js exportOptions: {fileName: export_ Moment().format("YYYY-MM-DD"),preventInjection: false,mso…...

python打卡day16
NumPy 数组基础 因为前天说了shap,这里涉及到数据形状尺寸问题,所以需要在这一节说清楚,后续的神经网络我们将要和他天天打交道。 知识点: numpy数组的创建:简单创建、随机创建、遍历、运算numpy数组的索引:…...

Redis 学习笔记 5:分布式锁
Redis 学习笔记 5:分布式锁 在前文中学习了如何基于 Redis 创建一个简单的分布式锁。虽然在大多数情况下这个锁已经可以满足需要,但其依然存在以下缺陷: 事实上一般而言,我们可以直接使用 Redisson 提供的分布式锁而非自己创建。…...

游戏开发实战(一):Python复刻「崩坏星穹铁道」嗷呜嗷呜事务所---源码级解析该小游戏背后的算法与设计模式【纯原创】
文章目录 奇美拉项目游戏规则奇美拉(Chimeras)档案领队成员 结果展示: 奇美拉项目 由于项目工程较大,并且我打算把我的思考过程和实现过程中踩过的坑都分享一下,因此会分3-4篇博文详细讲解本项目。本文首先介绍下游戏规则并给出奇美拉档案。…...
VS2017编译librdkafka 2.1.0
VS2017编译librdkafka 2.1.0 本篇是 Windows系统编译Qt使用的kafka(librdkafka)系列中的其中一篇,编译librdkafka整体步骤大家可以参考: Windows系统编译Qt使用的kafka(librdkafka) 由于项目需要,使用kafka,故自己编译了一次,编译的过程,踩了太多的坑了,特写了本篇…...

02- 浏览器运行原理
文章目录 1. 网页的解析过程浏览器内核 2. 浏览器渲染流程2.1 解析html2.2 生成css规则2.3 构建render tree2.4 布局(Layout)2.5 绘制(Paint) 3. 回流和重绘3.1 回流reflow(1)理解:(2)出现情况 3.2 重绘repaint&#x…...
Reactor模型详解与C++实现
Reactor模型详解与C实现 一、Reactor模型核心思想 Reactor模式是一种事件驱动的并发处理模型,核心通过同步I/O多路复用实现对多个I/O源的监听,当有事件触发时,派发给对应处理器进行非阻塞处理。 关键特征: 非阻塞I/Oÿ…...
人工智能重塑医疗健康:从辅助诊断到个性化治疗的全方位变革
人工智能正在以前所未有的速度改变着医疗健康领域,从影像诊断到药物研发,从医院管理到远程医疗,AI 技术已渗透到医疗服务的各个环节。本文将深入探讨人工智能如何赋能医疗健康产业,分析其在医学影像、临床决策、药物研发、个性化医…...

移除链表元素数据结构oj题(力扣题206)
目录 题目描述: 题目解读(分析) 解决代码 题目描述: 给你一个链表的头节点 head 和一个整数 val ,请你删除链表中所有满足 Node.val val 的节点,并返回 新的头节点 。 题目解读(分析&#…...
学习记录:DAY29
项目开发日志:技术实践与成长之路 前言 回顾这几天的状态,热情总是比我想象中更快被消耗完。比起茫然徘徊的小丑,我更希望自己是对着风车冲锋的疯子。 今天继续深入项目的实际业务。 状态好点的时候,再看自己EMO时写的东西&…...

OpenTelemetry 从入门到精通
快速入门 OpenTelemetry 是一个可观测性框架和工具包, 旨在创建和管理遥测数据,如链路、 指标和日志。 重要的是,OpenTelemetry 是供应商和工具无关的,这意味着它可以与各种可观测性后端一起使用, 包括 Jaeger 和 Pro…...
数学复习笔记 17
前言 复盘泰勒公式,极限四则运算,洛必达,拉格朗日。 1.27 因为是复习泰勒公式,所以就算有别的方法,我也硬是要用泰勒公式。就是为了记一下泰勒公式。泰勒公式确实是能做,但是做的我非常非常难受。公式确…...
C语言:在操作系统中,链表有什么应用?
在操作系统中,链表是一种重要的数据结构,凭借其灵活的内存管理和高效的插入/删除特性,被广泛应用于多个核心模块。以下是其主要应用场景及详细说明: 1. 内存管理:空闲内存块管理 应用场景:操作系统需要管…...
解锁MySQL性能调优:高级SQL技巧实战指南
高级SQL技巧:解锁MySQL性能调优的终极指南 开篇 当前,随着业务系统的复杂化和数据量的爆炸式增长,数据库性能调优成为了技术人员面临的核心挑战之一。尤其是在高并发、大数据量的场景下,SQL 查询的性能直接影响到整个系统的响应…...
裸金属服务器和云服务器之间的差别
裸金属服务器能够直接在硬件上运行,不需要额外的虚化层,让每个应用程序或者是服务都能够在实际的硬件上运行,不需要和其他虚拟服务器来共享资源;而云服务器作为一种虚拟服务器,是通过虚拟化技术为企业提供一个独立的计…...
WebSocket实时双向通信:从基础到实战
一、WebSocket 基础概念 1. 什么是 WebSocket? 双向通信协议:与 HTTP 的单向请求不同,WebSocket 支持服务端和客户端实时双向通信。 低延迟:适用于聊天室、实时数据推送、在线游戏等场景。 协议标识:ws://ÿ…...

【免杀】C2免杀技术(六)进程镂空(傀儡进程)
一、技术定位与核心思想 进程镂空(Process Hollowing)属于 MITRE ATT&CK 中 T1055.012 子技术:先创建一个合法进程并挂起,随后把其主模块从内存“掏空”并替换为恶意映像,最后恢复线程执行,从而让…...
ETL数据集成产品选型需要关注哪些方面?
ETL(Extract,Transform,Load)工具作为数据仓库和数据分析流程中的关键环节,其选型对于企业的数据战略实施有着深远的影响。谷云科技在 ETL 领域耕耘多年,通过自身产品的实践应用,对 ETL 产品选型…...

Eclipse Java 开发调优:如何让 Eclipse 运行更快?
Eclipse Java 开发调优:如何让 Eclipse 运行更快? 在 Java 开发领域,Eclipse 是一款被广泛使用的集成开发环境(IDE)。然而,随着项目的日益庞大和复杂,Eclipse 的运行速度可能会逐渐变慢&#x…...

彻底理解事件循环(Event Loop):从单线程到异步世界的桥梁
关于事件循环被问了很多次,也遇到过很多次,一直没有系统整理,网上搜的,基本明白但总感觉不够透彻,最后,自己动手,丰衣足食,哈哈 一、为什么需要事件循环?—— 单线程的困…...
java加强 -stream流
Stream流是jdk8开始新增的一套api,可以用于操作集合或数组的内容。 Stream流大量的结合了Lambda的语法风格来编程,功能强大,性能高效,代码简洁,可读性好。 体验Stream流 把集合中所有以三开头并且三个字的元素存储到…...
Vue百日学习计划Day33-35天详细计划-Gemini版
总目标: 在 Day 33-35 理解 Vue 组件从创建到销毁的完整生命周期,熟练掌握 Composition API 中主要的生命周期钩子,并知道在不同阶段执行哪些操作。 所需资源: Vue 3 官方文档 (生命周期钩子): https://cn.vuejs.org/guide/essentials/lifecycle.html你…...