【二叉树进阶】--- 前中后序遍历非递归

2024-08-30 23:12

本文主要是介绍【二叉树进阶】--- 前中后序遍历非递归,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

 Welcome to 9ilk's Code World

       

(๑•́ ₃ •̀๑) 个人主页:       9ilk

(๑•́ ₃ •̀๑) 文章专栏:     算法Journey 


本篇博客我们将来了解有关二叉树前中后序遍历的非递归版本。


🏠 前序遍历

要迭代非递归实现二叉树的前序遍历,首先还是要借助递归的类似思想,只需要把结点存在栈中,方便类似递归回退时取出父路径结点。跟这里不同的是,我们把一颗二叉树分为两个部分:1. 左路结点 2. 左路结点的右子树。

  • 我们先访问左路结点将他们依次入栈,再依次访问左路结点的右子树。
  • 访问左路结点的右子树只需要我们从栈里面取出左路结点,由于右子树又可以分为左路结点和右子树,所以我们以循环子问题的思想访问左路结点的右子树。

参考代码:

class Solution {
public:vector<int> ret;vector<int> preorderTraversal(TreeNode* root){TreeNode* cur = root;stack<TreeNode*> st;while(cur || !st.empty()){//访问左路节点while(cur)   {st.push(cur);ret.push_back(cur->val)cur = cur->left;}//从栈中取左路节点依次访问他们右子树TreeNode* top = st.top();st.pop();cur = top->right;    }return ret;            }
};

🏠 中序遍历

中序跟前序其实思路完全一致,只是访问根的时机不同。中序是对左路结点时不能先访问,而是先依次入栈,入栈完左路结点后再访问这些左路结点,再依次访问他们各自的右子树。

参考代码:

class Solution {
public:vector<int> ret;vector<int> inorderTraversal(TreeNode* root) {TreeNode* cur = root;stack<TreeNode*> st;while(cur || !st.empty()){//访问左路节点while(cur)   {st.push(cur); //先入栈不访问cur = cur->left;}TreeNode* top = st.top();ret.push_back(top->val);//访问st.pop();cur = top->right;    //访问右子树}return ret;     }
};

🏠 后序遍历

后序跟前序的思路也是完全一致,毕竟模拟的是递归展开过程,只不过后序是左子树-右子树-根,最后再访问根结点,也就是说要左右子树都访问完之后才能访问根并出栈。

TreeNode* top = st.top();
if(top->right == nullptr)
{ret.push_back(top->val);st.pop();
}
else //访问右子树
{cur = top->right;
}

当我们还面临一个问题当取到左路结点的右子树时,我们需要想办法标记判断右子树是否已经访问过了,如果访问过就直接访问根,没有访问过就访问右子树。因此我们上述逻辑访问右子树时不知道是否已经访问过右子树,处理完右子树后上一层的结点没有pop掉再次判断,因此会陷入循环。

解决方法是用一个prev变量来记录上一个访问的结点。但我们需要明白以下逻辑:

1. 取到一个左路结点时左子树已经访问过了

2.如果左路节点右子树不为空,右子树没有访问,那么上一个访问节点是左子树的根,此时需要访问右子树。

3.如果左路节点右子树不为空,右子树已经访问过,那么上一个访问节点应该是右子树的根,此时需要访问根

4.如果左路节点右子树为空,此时说明左子树已经访问,右子树不需要访问,此时需要访问根。

参考代码:

class Solution {
public:vector<int> ret;vector<int> postorderTraversal(TreeNode* root) {TreeNode* cur = root;stack<TreeNode*> st;TreeNode* pre = nullptr;while(cur || !st.empty()){//访问左路节点while(cur)   {st.push(cur);cur = cur->left;}// 这时代表左子树已经访问过了TreeNode* top = st.top();if(top->right == nullptr || top->right == pre)右子树为空或右子树已经访问完才访问根{ret.push_back(top->val);st.pop();   pre = top;}elsecur = top->right;    //右子树没访问再循环子问题访问右子树}return ret;     }
};

完。

这篇关于【二叉树进阶】--- 前中后序遍历非递归的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1122202

相关文章

Spring Boot + MyBatis Plus 高效开发实战从入门到进阶优化(推荐)

《SpringBoot+MyBatisPlus高效开发实战从入门到进阶优化(推荐)》本文将详细介绍SpringBoot+MyBatisPlus的完整开发流程,并深入剖析分页查询、批量操作、动... 目录Spring Boot + MyBATis Plus 高效开发实战:从入门到进阶优化1. MyBatis

Jackson库进行JSON 序列化时遇到了无限递归(Infinite Recursion)的问题及解决方案

《Jackson库进行JSON序列化时遇到了无限递归(InfiniteRecursion)的问题及解决方案》使用Jackson库进行JSON序列化时遇到了无限递归(InfiniteRecursi... 目录解决方案‌1. 使用 @jsonIgnore 忽略一个方向的引用2. 使用 @JsonManagedR

Java进阶学习之如何开启远程调式

《Java进阶学习之如何开启远程调式》Java开发中的远程调试是一项至关重要的技能,特别是在处理生产环境的问题或者协作开发时,:本文主要介绍Java进阶学习之如何开启远程调式的相关资料,需要的朋友... 目录概述Java远程调试的开启与底层原理开启Java远程调试底层原理JVM参数总结&nbsMbKKXJx

MySQL进阶之路索引失效的11种情况详析

《MySQL进阶之路索引失效的11种情况详析》:本文主要介绍MySQL查询优化中的11种常见情况,包括索引的使用和优化策略,通过这些策略,开发者可以显著提升查询性能,需要的朋友可以参考下... 目录前言图示1. 使用不等式操作符(!=, <, >)2. 使用 OR 连接多个条件3. 对索引字段进行计算操作4

Rust中的BoxT之堆上的数据与递归类型详解

《Rust中的BoxT之堆上的数据与递归类型详解》本文介绍了Rust中的BoxT类型,包括其在堆与栈之间的内存分配,性能优势,以及如何利用BoxT来实现递归类型和处理大小未知类型,通过BoxT,Rus... 目录1. Box<T> 的基础知识1.1 堆与栈的分工1.2 性能优势2.1 递归类型的问题2.2

JavaScript中的reduce方法执行过程、使用场景及进阶用法

《JavaScript中的reduce方法执行过程、使用场景及进阶用法》:本文主要介绍JavaScript中的reduce方法执行过程、使用场景及进阶用法的相关资料,reduce是JavaScri... 目录1. 什么是reduce2. reduce语法2.1 语法2.2 参数说明3. reduce执行过程

C++中使用vector存储并遍历数据的基本步骤

《C++中使用vector存储并遍历数据的基本步骤》C++标准模板库(STL)提供了多种容器类型,包括顺序容器、关联容器、无序关联容器和容器适配器,每种容器都有其特定的用途和特性,:本文主要介绍C... 目录(1)容器及简要描述‌php顺序容器‌‌关联容器‌‌无序关联容器‌(基于哈希表):‌容器适配器‌:(

Python进阶之Excel基本操作介绍

《Python进阶之Excel基本操作介绍》在现实中,很多工作都需要与数据打交道,Excel作为常用的数据处理工具,一直备受人们的青睐,本文主要为大家介绍了一些Python中Excel的基本操作,希望... 目录概述写入使用 xlwt使用 XlsxWriter读取修改概述在现实中,很多工作都需要与数据打交

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java进阶13讲__第12讲_1/2

多线程、线程池 1.  线程概念 1.1  什么是线程 1.2  线程的好处 2.   创建线程的三种方式 注意事项 2.1  继承Thread类 2.1.1 认识  2.1.2  编码实现  package cn.hdc.oop10.Thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory