二叉树的前序、中序、后序遍历 -- 非递归方式实现

目录

前序遍历:根 左 右

实现思想:

需要创建一个栈和一个vector容器

栈用来保存最左路径的节点

vector用来保存遍历的数据

1.首先用一个while循环将二叉树的最左路径节点全部压入栈内,同时由于前序遍历的特性,也将该节点的值同时压入vector容器内

2.循环完成后,栈顶元素为二叉树的最左节点。将该节点获取为当前节点后,对栈顶元素进行移除。

3.访问当前节点的右子树。

4.重复循环,直到当前节点为空,或者栈为空。

实现过程:

cpp 复制代码
class Solution
{
public:
	//前序遍历:根 左 右(非递归)
	vector<int> preorderTraversal(TreeNode* root)
	{
		stack<TreeNode*> st;
		vector<int> vec;
		TreeNode* cur = root;
		while (cur || !st.empty())
		{
			while (cur)
			{
				//将二叉树最左路径压入栈内
				st.push(cur);
				//将最左节点的值压入vector内
				vec.push_back(cur->val);
				cur = cur->left;
			}
			//获取最左节点
			cur = st.top();
			//将最左节点从栈内移除
			st.pop();

			//访问右子树
			cur = cur->right;
		}
		return vec;
	}
};

中序遍历:左 根 右

实现思想:

和前序遍历十分类似,不同之处在于将节点值压入vector的位置不一样。

前序遍历是在将二叉树最左路径节点压入栈的同时,也将该节点的值压入vector。

但是中序遍历是要先从最左节点开始进行,因此将节点值压入vector的操作换到第二步完成。

创建一个栈和一个vector容器

栈用来保存最左路径的节点

vector用来保存遍历的数据

1.首先用一个while循环将二叉树的最左路径节点全部压入栈内。

2.循环完成后,栈顶元素为二叉树的最左节点。将该节点获取为当前节点后,对栈顶元素进行移除,并将该节点的值压入vector容器

3.访问当前节点的右子树。

4.重复循环,直到当前节点为空,或者栈为空。

实现过程:

cpp 复制代码
class Solution
{
public:
	//中序遍历:左 根 右(非递归)
	vector<int> inorderTraversal(TreeNode* root)
	{
		stack<TreeNode*> st;
		vector<int> vec;
		TreeNode* cur = root;
		while (cur || !st.empty())
		{
			while (cur)
			{
				//将二叉树最左路径压入栈内
				st.push(cur);
				cur = cur->left;
			}
			//获取最左节点
			cur = st.top();
			//将最左节点从栈内移除
			st.pop();
			//将最左节点的值压入vector内
			vec.push_back(cur->val);

			//访问右子树
			cur = cur->right;
		}
		return vec;
	}
};

后序遍历:左 右 根

实现思想:

创建一个栈和一个vector容器

栈用来保存最左路径的节点

vector用来保存遍历的数据

创建两个指针

一个指向当前节点、一个指向上一栈顶节点

1.首先用一个while循环将二叉树的最左路径节点全部压入栈内。

2.循环完成后,栈顶元素为二叉树的最左节点。将栈顶元素获取为top节点。

3.判断top节点是否可以压入vector,条件为top节点的右子树为空,或者top节点的右子树已经被压入vector。满足条件后将该top节点压入vector,同时将prev更新。

3.如果不满足条件,则访问top节点的右子树。

4.重复循环,直到当前节点为空,或者栈为空。

实现过程:

cpp 复制代码
class Solution
{
public:
	//后序遍历:左 右 根(非递归)
	vector<int> postorderTraversal(TreeNode* root)
	{
		stack<TreeNode*> st;
		TreeNode* cur = root;
		TreeNode* prev;
		vector<int> vec;
		while (cur || !st.empty())
		{
			while (cur)
			{
				st.push(cur);
				cur = cur->left;
			}
			//获取栈顶元素
			TreeNode* top = st.top();
			//判断当前的节点是否可以遍历
			if (top->right == nullptr || top->right == prev)
			{
				vec.push_back(top->val);
				st.pop();
				prev = top;
			}
			else
				//访问右子树
				cur = top->right;
		}
		return vec;
	}
};
相关推荐
MiyamiKK575 分钟前
leetcode_字符串 409. 最长回文串
数据结构·算法·leetcode
半盏茶香26 分钟前
扬帆数据结构算法之雅舟航程,漫步C++幽谷——LeetCode刷题之移除链表元素、反转链表、找中间节点、合并有序链表、链表的回文结构
数据结构·c++·算法
哎呦,帅小伙哦34 分钟前
Effective C++ 规则41:了解隐式接口和编译期多态
c++·effective c++
CodeJourney.1 小时前
小型分布式发电项目优化设计方案
算法
DARLING Zero two♡1 小时前
【初阶数据结构】逆流的回环链桥:双链表
c语言·数据结构·c++·链表·双链表
9毫米的幻想1 小时前
【Linux系统】—— 编译器 gcc/g++ 的使用
linux·运维·服务器·c语言·c++
带多刺的玫瑰1 小时前
Leecode刷题C语言之从栈中取出K个硬币的最大面积和
数据结构·算法·图论
Cando学算法1 小时前
Codeforces Round 1000 (Div. 2)(前三题)
数据结构·c++·算法
薯条不要番茄酱1 小时前
【动态规划】落花人独立,微雨燕双飞 - 8. 01背包问题
算法·动态规划
小林熬夜学编程1 小时前
【Python】第三弹---编程基础进阶:掌握输入输出与运算符的全面指南
开发语言·python·算法