代码随想录算法训练营day14|二叉树的递归遍历、二叉树的迭代遍历、二叉树的统一迭代法

二叉树的递归遍历

首先需要明确的一点是,前序中序和后序在二叉树的递归遍历中的区别仅在于递归函数中操作的顺序,前序是在遍历一个节点的左右子树前进行操作,中序是在遍历一个节点的左子树后进行操作再遍历右子树,而后序是在遍历完左右子树再进行操作。所以在递归函数中,仅是顺序的差别。前序中序和后序可简化为中左右、左中右和左右中,此外,写递归函数时,需要明确三点:

  • 确定递归函数的参数和返回值
  • 确定终止条件
  • 确定单层递归的逻辑

        在对二叉树的递归遍历中,递归函数需要传入树节点TreeNode * cur和vec数组进行操作,无实际返回值,所以函数为void,当递归操作时,若节点为空则返回,对每层递归的逻辑以前序遍历来说,先讲当前节点的val值传入数组vec中,之后遍历左子树,再之后遍历右子树。

代码随想录 (programmercarl.com)二叉树的递归遍历icon-default.png?t=N7T8https://programmercarl.com/%E4%BA%8C%E5%8F%89%E6%A0%91%E7%9A%84%E9%80%92%E5%BD%92%E9%81%8D%E5%8E%86.html#%E6%80%9D%E8%B7%AF看到递归就晕?带你理解递归的本质!_哔哩哔哩_bilibiliicon-default.png?t=N7T8https://www.bilibili.com/video/BV1UD4y1Y769/?spm_id_from=333.880.my_history.page.click&vd_source=fc4a6e70e3a87b7ea67c2024e326e7c5

前序遍历递归函数如下所示,

void traversal(TreeNode * cur, vector<int> & vec){//创建前序递归遍历函数if(cur == nullptr)//当当前遍历节点为空时,返回return;vec.push_back(cur->val);//中 将当前遍历节点的val值存入vec数组 *操作traversal(cur->left,vec);//遍历左子树traversal(cur->right,vec);//遍历右子树}void traversal(TreeNode * cur, vector<int> & vec){//创建中序递归遍历函数if(cur == nullptr)//当当前遍历节点为空时,返回return;traversal(cur->left,vec);//遍历左子树vec.push_back(cur->val);//中 将当前遍历节点的val值存入vec数组 *操作traversal(cur->right,vec);//遍历右子树}void traversal(TreeNode * cur, vector<int> & vec){//创建后序递归遍历函数if(cur == nullptr)//当当前遍历节点为空时,返回return;traversal(cur->left,vec);//遍历左子树traversal(cur->right,vec);//遍历右子树vec.push_back(cur->val);//中 将当前遍历节点的val值存入vec数组 *操作}

对于主函数体,只需传入节点,在函数体中创建一个ans数组,并传入traversal函数中。

vector<int> preorderTraversal(TreeNode* root) {vector<int> ans;traversal(root,ans);return ans;}

递归遍历树节点的时间复杂度和空间复杂度均为O(n)。

二叉树的迭代遍历

前序遍历

        考虑前序递归遍历的过程,若操作为打印,则先将自身val值打印后,对所有左子树打印,然后打印右子树,考虑使用栈来模拟前序遍历的过程(也是用栈来模拟递归的过程),由于栈是先入后出的数据结构,前序遍历结果左子树节点在右子树节点前,所以需先对右子树中节点入栈,再对左子树中节点入栈。然后是大致流程,创建结果数组ans,先判断根节点是否为空,若为空直接返回结果数组,否则将根节点入栈,在一个while循环中实现二叉树的迭代前序遍历,由于栈模拟的递归过程,当栈为空时,也就意味着递归结束,即我们遍历结果结果结束。所以while的循环条件为stack.size()!=0,在循环体内,创建一个指针指向当前节点,将其val值加入ans数组,之后再弹出栈,(第一次循环的pop为根节点,即第一次前序遍历要对根节点进行操作,之后因为我们是对栈是先加入右子树节点后加入左子树节点,所以会先对节点的左子树进行操作后对节点的右子树进行操作,完成前序遍历的过程),之后判断右子树的存在,若存在,将其入栈,判断左子树的存在,存在则将其入栈,这就是循环体内内容。

class Solution {
public:vector<int> preorderTraversal(TreeNode* root) {stack<TreeNode*> stack;vector<int> ans;if(root == nullptr){//当根节点为nullptr,直接返回ansreturn ans;}stack.push(root);while(stack.size()!=0){//栈不为空则循环TreeNode*cur = stack.top();//创建cur指针指向栈顶地址ans.push_back(cur->val);//将栈顶地址的值存入ans数组stack.pop();//出栈if(cur->right)//判断左右子树存在的情况,若存在,将其存入栈中,注意顺序,先右后左                //出栈则为先左后右stack.push(cur->right);if(cur->left)stack.push(cur->left);}return ans;}
};

该算法的时间复杂度和空间复杂度均为O(n)。

中序遍历

        在遍历二叉树的节点时,创建遍历指针cur,先遍历左子树,依次将所有左节点全部入栈,当当前节点的左节点为空时,弹出栈中元素给遍历指针cur,并将cur指向的val传入数组中,完成中的操作,令cur = cur->right,并继续进行循环。算法的时间和空间复杂度均为O(n)。

class Solution {
public:// 定义一个函数,用于执行中序遍历vector<int> inorderTraversal(TreeNode* root) {// 创建一个空向量 ans,用于存储遍历的结果vector<int> ans;// 创建一个空栈 st,用于辅助遍历stack<TreeNode*> st;// 创建一个指针 cur,初始指向根节点TreeNode* cur = root;// 当 cur 指向的节点不为空或者栈 st 不为空时,进行循环while (cur != nullptr || !st.empty()) {// 如果 cur 指向的节点不为空,则将其压入栈 st 中,并移动 cur 指针到其左子节点if (cur != nullptr) {st.push(cur);cur = cur->left;}// 如果 cur 指向的节点为空,说明左子节点已经访问完毕,此时从栈 st 中弹出最上面的节点else {cur = st.top(); // 获取栈顶节点st.pop(); // 弹出栈顶节点// 将弹出的节点的值添加到 ans 向量中ans.push_back(cur->val);// 移动 cur 指针到其右子节点cur = cur->right;}}// 循环结束后,返回存储遍历结果的 ans 向量return ans;}
};

后序遍历

        考虑前序遍历和后序遍历的相似之处,调整一下前序遍历中左右节点的入栈顺序,让中左右变为中右左,之后再通过一个reverse即能实现后序遍历。

调整

//前序
if(cur->right)//判断左右子树存在的情况,若存在,将其存入栈中,注意顺序,先右后左                //出栈则为先左后右
stack.push(cur->right);
if(cur->left)
stack.push(cur->left);//后序
if(cur->left)
stack.push(cur->left);
if(cur->right)
stack.push(cur->right);

整体代码

class Solution {
public:vector<int> postorderTraversal(TreeNode* root) {stack<TreeNode*> stack;vector<int> ans;if(root == nullptr){//当根节点为nullptr,直接返回ansreturn ans;}stack.push(root);while(stack.size()!=0){//栈不为空则循环TreeNode*cur = stack.top();//创建cur指针指向栈顶地址ans.push_back(cur->val);//将栈顶地址的值存入ans数组stack.pop();//出栈if(cur->left)stack.push(cur->left);if(cur->right)stack.push(cur->right);}reverse(ans.begin(),ans.end());//反转return ans;}
};

二叉树的统一迭代法

周末再看看

代码随想录 (programmercarl.com)二叉树的统一迭代法icon-default.png?t=N7T8https://programmercarl.com/%E4%BA%8C%E5%8F%89%E6%A0%91%E7%9A%84%E7%BB%9F%E4%B8%80%E8%BF%AD%E4%BB%A3%E6%B3%95.html#%E6%80%9D%E8%B7%AF

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

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

相关文章

C++算术运算和自增自减运算

一 引言 表示运算的符号称为运算符。 算术运算&#xff1b; 比较运算&#xff1b; 逻辑运算&#xff1b; 位运算&#xff1b; 1 算术运算 算术运算包括加、减、乘、除、乘方、指数、对数、三角函数、求余函数&#xff0c;这些都是算术运算。 C中用、-、*、/、%分别表示加、减…

《当微服务遇上Ribbon:一场负载均衡的华丽舞会》

在微服务的厨房里&#xff0c;如何确保每一道服务都恰到好处&#xff1f;揭秘Spring Cloud Ribbon如何像大厨一样精心调配资源&#xff0c;让负载均衡变得像烹饪艺术一样简单&#xff01; 文章目录 Spring Cloud Ribbon 详解1. 引言微服务架构中的负载均衡需求Spring Cloud Rib…

【算法实战】每日一题:设计一个算法,用最少数量的矩形覆盖一系列宽度为d、高度为w的矩形,且使用矩形不能超出边界

题目 设计一个算法&#xff0c;用最少数量的矩形覆盖一系列宽度为d、高度为w的矩形建筑物侧墙&#xff0c;且矩形不能超出边界。 核心思路 考虑这种结构 前面递增后面一个与前面的某个高度一致&#xff0c;这时候考虑最下面的覆盖&#xff08;即都是从最下面向上覆盖&#…

redis数据类型set,zset

华子目录 Set结构图相关命令sdiff key1 [key2]sdiffstore destination key1 [key2...]sinter key1 [key2...]sinterstore destination key1 [key2...]sunion key1 [key2...]sunionstore destination key1 [key2...]smove source destination memberspop key [count]sscan key c…

Java GC问题排查的一些个人总结和问题复盘

个人博客 Java GC问题排查的一些个人总结和问题复盘 | iwts’s blog 是否存在GC问题判断指标 有的比较明显&#xff0c;比如发布上线后内存直接就起飞了&#xff0c;这种也是比较好排查的&#xff0c;也是最多的。如果单纯从优化角度&#xff0c;看当前应用是否需要优化&…

Unity实现首行缩进两个字符

效果 在Unity中如果想实现首行缩进两个字符&#xff0c;你会发现按空格是没法实现的。 实现原理&#xff1a;用空白的透明的字替代原来的位置。 代码&#xff1a; <color#FFFFFF00>XXX</color> 赶紧去试试吧&#xff01;

备战秋招—模拟版图面试题来了

随着暑期的脚步逐渐临近&#xff0c;电子工程和集成电路设计领域的毕业生们&#xff0c;也将迎来了另一个求职的黄金期——秋招。我们总说机会是留给有准备的人。对于有志于投身于模拟版图设计的学子们来说&#xff0c;为了在众多求职者中脱颖而出&#xff0c;充分备战模拟版图…

C# 工商银行缺少infosecapiLib.infosec

搜索Tlbimp.exe 这里使用4.8.1下的处理&#xff0c;以管理员身份打开powershell cd "C:\Program Files (x86)\Microsoft SDKs\Windows\v10.0A\bin\NETFX 4.8.1 Tools".\TlbImp.exe "G:\CSharp\icbc-api-sdk-cop-c#\sdk-cop\sdk-cop\dll\infosecapi.dll" …

PCIe协议之-DLLP详解

✨前言&#xff1a; &#x1f31f;数据链路层的功能 数据链路层将从物理层中获得报文&#xff0c; 并将其传递给事务层&#xff1b; 同时接收事务层的报文&#xff0c; 并将其转发到物理层; 核心的功能有以下三点 1.保证TLP在 PCIe 链路中的正确传递; 2.数据链路层使用了容错…

杀死那个进程

一、场景 eclipse在启动tomcat时&#xff0c;出现端口被占用的情况。我寻思着“任务管理器”没出现相应程序在跑啊。 1.1问题&#xff1a;端口和进程的关系 端口和进程之间存在着一种关系&#xff0c;端口是一个逻辑概念&#xff0c;它用于标识网络通信中的一个终点&#xff0…

SEC突发:以太坊ETF大概率获批

美国证监会大概率批准以太坊现货ETF。 5月20日&#xff0c;据外媒CoinDesk报道&#xff0c;知情人士透露&#xff0c;美国SEC周一要求证券交易所更新以太坊现货ETF的19b-4备案文件。19b-4备案文件是一种表格&#xff0c;用于向SEC通报允许基金在交易所交易的规则变更。 三位消息…

利用cherry pick巧妙地将某次提交单独合并到其他分支

0. 引言 最近在进行系统的多版本并行开发&#xff0c;涉及一些共有基础功能提交时就遇到了麻烦&#xff0c;一份代码需要向多个版本分支进行同步&#xff0c;以保证多版本都能有更新该基础功能。 多次对比提交的方式显然会带来巨大的工作量。但实际上我们可以通过git的cherry…

「Python Socket超能力:网络世界的隐形斗篷!」

Hi&#xff0c;我是阿佑&#xff0c;今天将带领大家揭开Python Socket编程的神秘面纱&#xff0c;赋予我们的网络应用隐形斗篷般的超能力&#xff01; 深入探讨Socket编程的革命性力量&#xff0c;教你如何用Python的Socket模块来构建强大的网络应用。从简单的HTTP服务器到复杂…

MagicLens:新一代图像搜索技术和产品形态

MagicLens&#xff1a;Self-Supervised Image Retrieval with Open-Ended Instructions MagicLens: 自监督图像检索与开放式指令 作者&#xff1a;Kai Zhang&#xff0c; Yi Luan&#xff0c; Hexiang Hu&#xff0c; Kenton Lee&#xff0c; Siyuan Qiao&#xff0c; Wenhu …

在VS Code中进行Java的单元测试

在VS Code中可以使用 Test Runner for Java扩展进行Java的测试执行和调试。 Test Runner for Java的功能 Test Runner for Java 结合 Language Support for Java by Red Hat 和 Debugger for Java这两个插件提供如下功能&#xff1a; 运行测试&#xff1a; Test Runner for …

QT学习(20):QStyle和自定义样式

QStyle 样式&#xff08;继承自QStyle类&#xff09;代表控件的绘制并封装GUI的外观。QStyle是一个封装了GUI外观的抽象基类。Qt使用QStyle去执行几乎所有的内置控件的绘制&#xff0c;确保控件外观和原生控件风格风格相同。 class Q_WIDGETS_EXPORT QStyle : public QObject{…

【OpenCV】图像通道合并与分离,ROI

介绍可以实现图像通道合并与分离的API&#xff0c;这只是一种方式&#xff0c;后续还会介绍其他的合并与分离方法&#xff0c;以及ROI区域截取的方法。相关API&#xff1a; split() merge() Mat对象() 代码&#xff1a; #include "iostream" #include "ope…

MySQL进阶之(九)数据库的设计规范

九、数据库的设计规范 9.1 范式的概念9.1.1 范式概述9.1.2 键和相关属性 9.2 常见的范式9.2.1 第一范式9.2.2 第二范式9.2.3 第三范式9.2.4 第四范式9.2.5 第五范式&#xff08;域键范式&#xff09; 9.3 反范式化9.3.1 概述9.3.2 举例9.3.3 反范式化新问题9.3.4 通用场景 9.4 …

18 - grace数据处理 - 补充 - 地下水储量计算过程分解 - 地表水储量变化Glads水文数据处理

18 - grace数据处理 - 补充 - 地下水储量计算过程分解 - 地表水储量变化 0 引言1 Grace陆地水储量过程整合0 引言 由水量平衡方程可以将地下水储量的计算过程分解为3个部分,第一部分计算陆地水储量变化、第二部分计算地表水储量变化、第三部分计算地下水储量变化。本篇简单介绍…

2024.05.28学习记录

1. 小林coding 计网复习 2.代码随想录刷题. 图论.和复习数组.链表 3.rosebush完成select组件