news 2026/5/11 5:41:53

二叉树的遍历(全面总结)

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
二叉树的遍历(全面总结)

目录

二叉树遍历分为两大类:

二叉树节点结构

深度优先遍历(DFS)

二叉树前序递归遍历

二叉树前序非递归遍历

二叉树中序递归遍历

二叉树中序非递归遍历

二叉树后序递归遍历

二叉树后序非递归遍历

广度优先遍历(BFS)

二叉树的层序遍历


二叉树遍历分为两大类:

深度优先遍历(DFS):前序、中序、后序

广度优先遍历(BFS):层次遍历

掌握二叉树的遍历是理解和操作二叉树的基础。递归方法简洁易懂,迭代方法效率更高且避免栈溢出

二叉树节点结构

struct TreeNode { int val; TreeNode* left; TreeNode* right; TreeNode() : val(0), left(nullptr), right(nullptr) {} TreeNode(int x) : val(x), left(nullptr), right(nullptr) {} TreeNode(int x, TreeNode* left, TreeNode* right) : val(x), left(left), right(right) {} };

深度优先遍历(DFS)

二叉树前序递归遍历

vector<int> preorderTraversal(TreeNode* root) { vector<int> result; preorder(root, result); return result; } void preorder(TreeNode* root, vector<int>& result) { if (!root) return; result.push_back(root->val); // 访问根 preorder(root->left, result); // 遍历左子树 preorder(root->right, result); // 遍历右子树 }

二叉树前序非递归遍历

思路:前序非递归遍历需要借助栈
1. 如果树为空,直接返回
2. 如果树非空:从根节点位置开始遍历,因为前序遍历规则:根节点、左子树、右子树
a. 沿着根节点一直往左走,将所经过路径中的节点依次入栈,并访问。
b. 取栈顶元素,该元素取到后,其左子树要么为空,要么已经遍历,可以直接遍历该节点,对于该节点,其左子树已经遍历,该节点也已经遍历,剩余其右子树没有遍历,将其左子树当成一棵新的树开始遍历,继续a

代码实现:

class Solution { public: vector<int> preorderTraversal(TreeNode* root) { vector<int> v; stack<TreeNode*> st; TreeNode* cur = root; while(!st.empty() || cur) { // 每次循环表示要开始访问一颗树了,先将一颗数的左路节点都入栈并访问节点 // 剩余左路节点的右子树还没访问 while(cur) { v.push_back(cur->val); st.push(cur); cur = cur->left; } // 取栈中的节点依次访问左路节点的右子树 TreeNode* top = st.top(); st.pop(); cur = top->right; } return v; } };

二叉树中序递归遍历

vector<int> inorderTraversal(TreeNode* root) { vector<int> result; inorder(root, result); return result; } void inorder(TreeNode* root, vector<int>& result) { if (!root) return; inorder(root->left, result); // 遍历左子树 result.push_back(root->val); // 访问根 inorder(root->right, result); // 遍历右子树 }

二叉树中序非递归遍历

思路:中序非递归遍历需要借助栈 1. 空树,直接返回 2. 如果树非空:从根节点位置开始遍历,但此时根节点不能遍历,因为中序遍历规则:左子树、根节点、右子树 a. 沿着根节点一直往左走,将所经过路径中的节点依次入栈 b. 取栈顶元素,该元素取到后,其左子树要么为空,要么已经遍历,可以直接遍历该节点,对于该节点其左子树已经遍历,该节点也已经遍历,剩余其右子树没有遍历,将其左子树当成一棵新的树开始遍历,继续a

代码实现:

class Solution { public: vector<int> inorderTraversal(TreeNode* root) { // 空树,直接返回 vector<int> vRet; if(nullptr == root) return vRet; TreeNode* pCur = root; stack<TreeNode*> s; while(pCur || !s.empty()) { // 找以pCur为根的二叉树最左侧的节点,并将所经路径中的节点入栈 while(pCur) { s.push(pCur); pCur = pCur->left; } pCur = s.top(); // pCur左子树为空,相当于左子树已经访问过了,可以直接访问以pCur为根的二叉树的根节点 vRet.push_back(pCur->val); s.pop(); // 以pCur为根的二叉树的左子树已经遍历完,根节点已经遍历, // 将pCur的右子树当成一棵二叉树来遍历 pCur = pCur->right; } return vRet; } };

二叉树后序递归遍历

vector<int> postorderTraversal(TreeNode* root) { vector<int> result; postorder(root, result); return result; } void postorder(TreeNode* root, vector<int>& result) { if (!root) return; postorder(root->left, result); // 遍历左子树 postorder(root->right, result); // 遍历右子树 result.push_back(root->val); // 访问根 }

二叉树后序非递归遍历

思路:后序非递归遍历需要借助栈 1. 空树,直接返回 2. 如果树非空:从根节点位置开始遍历,但此时根节点不能遍历,因为后序遍历规则:左子树、右子树、根节点 a. 沿着根节点一直往左走,将所经过路径中的节点依次入栈 b. 取栈顶元素,该元素取到后,其左子树要么为空,要么已经遍历,但是此时该节点不能遍历,除非其右子树不存在或者其右子树已经遍历,才可以遍历该节点.如果该节点右子树没有遍历,将其右子树作为一棵新的二叉树遍历,继续a

代码实现:

class Solution { public: vector<int> postorderTraversal(TreeNode* root) { // 空树直接返回 vector<int> vRet; if(nullptr == root) return vRet; TreeNode* pCur = root; TreeNode* pPrev = nullptr; stack<TreeNode*> s; while(pCur || !s.empty()) { // 找以pCur为根的二叉树最左侧的节点,并将所经路径中的节点入栈 while(pCur) { s.push(pCur); pCur = pCur->left; } TreeNode* pTop = s.top(); // pTop左子树已经访问 // 如果pTop的右子树是空,或者右子树已经访问过了,就可以访问pTop if(nullptr == pTop->right || pPrev == pTop->right) { vRet.push_back(pTop->val); s.pop(); // 将刚刚访问过的节点标记起来 pPrev = pTop; } else { // 如果右子树没有访问,将右子树当成一棵新的二叉树访问 pCur = pTop->right; } } return vRet; } };

广度优先遍历(BFS)

二叉树的层序遍历

层序遍历:从上到下,从左到右逐层访问二叉树的所有节点。通过这个性质,我们可以联想到队列的FIFO特性(保证了节点按入队顺序访问),先入队的节点先访问,符合层序要求

vector<vector<int>> levelOrder(TreeNode* root) { vector<vector<int>> result; if (!root) return result; queue<TreeNode*> q; q.push(root); while (!q.empty()) { int levelSize = q.size(); vector<int> level; for (int i = 0; i < levelSize; i++) { TreeNode* node = q.front(); q.pop(); level.push_back(node->val); if (node->left) q.push(node->left); if (node->right) q.push(node->right); } result.push_back(level); } return result; }
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/8 19:00:48

ArrayPool.Shared解说

NET 中频繁创建和销毁数组的情况下会导致垃圾回收器出现严重的内存压力&#xff0c;ArrayPool<T> 通过池化手段有效地降低了数组的分配和垃圾回收器的回收压力&#xff0c;同时 ArrayPool<T> 也是 MemoryPool<T> 和 PipeWriter、PipeReader 的底板。ArrayPoo…

作者头像 李华
网站建设 2026/5/1 9:10:00

VonaJS提供的读写分离,直观,优雅[特殊字符]

在VonaJS中实现读写分离&#xff0c;只需提供一组写数据源和一组读数据源。当用户访问后端 API 时&#xff0c;系统会按照规则自动选择写数据源或读数据源&#xff0c;访问相应的数据库&#xff0c;从而分摊压力&#xff0c;提升系统性能安装模块读写分离作为独立的模块提供&am…

作者头像 李华
网站建设 2026/5/1 9:08:02

鸿蒙6.0:AI与智能体框架(HMAF),重塑操作系统未来的核心密码

当用户说出“帮我规划带老人孩子的周末短途游”&#xff0c;系统便能自动整合行程、餐饮、景点资源生成完整方案&#xff1b;当驾车抵达加油站&#xff0c;车载系统自动识别油枪并完成人脸支付&#xff1b;当需要分析Excel数据&#xff0c;仅凭自然语言就能完成复杂报表生成——…

作者头像 李华
网站建设 2026/5/10 9:07:08

【往届均已成功见刊检索、早鸟优惠】第六届计算机网络安全与软件工程国际学术会议(CNSSE 2026)

第六届计算机网络安全与软件工程国际学术会议&#xff08;CNSSE 2026&#xff09;将于2026年2月6-8日在中国-青岛举行。CNSSE 2026专注于计算机网络安全、软件工程、信号处理、程序分析等领域&#xff0c;致力于搭建计算机领域学术资源共享平台&#xff0c;扩大国际科研学术合作…

作者头像 李华
网站建设 2026/5/9 21:36:55

你必须知道的TCP和UDP核心区别,快速搞懂这两大协议!

1. TCP (Transmission Control Protocol)概念TCP&#xff08;传输控制协议&#xff09;是一种面向连接的、可靠的传输协议。它负责将数据从源主机传输到目标主机&#xff0c;并确保数据的完整性、顺序和正确性。原理三次握手&#xff1a;在数据传输之前&#xff0c;TCP协议通过…

作者头像 李华
网站建设 2026/5/10 7:00:02

什么是智能体工程Agent Engineering?

文章介绍了智能体工程这一新兴领域&#xff0c;它是将不可靠的大模型系统转化为生产环境稳定应用的方法论。强调构建、测试、上线、观察、优化的循环过程&#xff0c;需要产品思维、工程能力和数据科学的配合。与传统开发不同&#xff0c;智能体工程将生产环境视为最佳学习场所…

作者头像 李华