news 2026/5/5 4:28:29

【LeetCode刷题日记】掌握二叉树遍历:栈实现的三种绝妙方法

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
【LeetCode刷题日记】掌握二叉树遍历:栈实现的三种绝妙方法

🔥个人主页:北极的代码(欢迎来访)
🎬作者简介:java后端学习者
❄️个人专栏:苍穹外卖日记,SSM框架深入,JavaWeb
命运的结局尽可永在,不屈的挑战却不可须臾或缺!

前言:

我们在前面提到了,递归的实现就是:每一次递归调用都会把函数的局部变量、参数值和返回地址等压入调用栈中,然后递归返回的时候,从栈顶弹出上一次递归的各项参数,所以这就是递归为什么可以返回上一层位置的原因。

此时大家应该知道我们用栈也可以是实现二叉树的前后中序遍历了。

摘要:

本文介绍了使用栈实现二叉树三种遍历方式的迭代算法。

前序遍历通过"根→右→左"的入栈顺序实现;

中序遍历需要指针辅助,先一路向左压栈再处理节点;

后序遍历采用"根→右→左"顺序处理后反转结果。

三种方法对比:前序遍历访问与处理顺序一致,无需指针;中序遍历需要指针跟踪未处理节点;后序遍历通过反转前序变体实现。这些方法通过显式栈替代递归隐式栈,均能达到O(n)时间复杂度,是面试常考的二叉树遍历实现方案。

前序遍历(根 → 左 → 右)

思路

  • 访问和处理顺序一致:先根节点 → 再左 → 再右。

  • 用栈来模拟递归:先压右孩子,再压左孩子,这样出栈时先左后右。

步骤

  1. 根节点入栈。

  2. 循环直到栈为空:

    • 弹出栈顶 → 处理(加入结果)。

    • 右孩子非空 → 入栈。

    • 左孩子非空 → 入栈。

java public List<Integer> preorderTraversal(TreeNode root) { List<Integer> result = new ArrayList<>(); if (root == null) return result; Stack<TreeNode> stack = new Stack<>(); stack.push(root); while (!stack.isEmpty()) { TreeNode node = stack.pop(); result.add(node.val); // 中 if (node.right != null) stack.push(node.right); // 右 if (node.left != null) stack.push(node.left); // 左 } return result; }

二、中序遍历(左 → 根 → 右)

为了解释清楚,我说明一下 刚刚在迭代的过程中,其实我们有两个操作:

  1. 处理:将元素放进result数组中
  2. 访问:遍历节点

分析一下为什么刚刚写的前序遍历的代码,不能和中序遍历通用呢,因为前序遍历的顺序是中左右,先访问的元素是中间节点,要处理的元素也是中间节点,所以刚刚才能写出相对简洁的代码,因为要访问的元素和要处理的元素顺序是一致的,都是中间节点。

那么再看看中序遍历,中序遍历是左中右,先访问的是二叉树顶部的节点,然后一层一层向下访问,直到到达树左面的最底部,再开始处理节点(也就是在把节点的数值放进result数组中),这就造成了处理顺序和访问顺序是不一致的。

那么在使用迭代法写中序遍历,就需要借用指针的遍历来帮助访问节点,栈则用来处理节点上的元素。

思路

  • 访问顺序:从根开始一直向左走到底。

  • 处理顺序:最左的子节点先处理,然后根,然后右子树。

  • 访问和处理顺序不一致→ 需要指针 + 栈。

步骤

  1. 指针cur指向根。

  2. 循环条件:cur != null或栈非空。

    • 如果cur != null:入栈 + 往左。

    • 否则(cur 为空):

      • 弹出栈顶(处理)。

      • 指针指向右孩子。

java public List<Integer> inorderTraversal(TreeNode root) { List<Integer> result = new ArrayList<>(); Stack<TreeNode> stack = new Stack<>(); TreeNode cur = root; while (cur != null || !stack.isEmpty()) { if (cur != null) { stack.push(cur); cur = cur.left; // 左 } else { cur = stack.pop(); result.add(cur.val); // 中 cur = cur.right; // 右 } } return result; }

三、后序遍历(左 → 右 → 根)

思路

  • 利用前序遍历的变体:

    • 前序:根 → 左 → 右(栈:先右后左)。

    • 变体:根 → 右 → 左(栈:先左后右)。

  • 最后将结果反转→ 得到左 → 右 → 根。

步骤

  1. 根入栈。

  2. 循环:

    • 弹出栈顶 → 加入结果。

    • 先压左孩子,再压右孩子(保证出栈顺序是“根→右→左”)。

  3. 结果反转。

java public List<Integer> postorderTraversal(TreeNode root) { List<Integer> result = new ArrayList<>(); if (root == null) return result; Stack<TreeNode> stack = new Stack<>(); stack.push(root); while (!stack.isEmpty()) { TreeNode node = stack.pop(); result.add(node.val); if (node.left != null) stack.push(node.left); if (node.right != null) stack.push(node.right); } Collections.reverse(result); return result; }

四、关键对比总结(面试常问)

遍历方式访问与处理顺序是否一致是否需要指针核心技巧
前序✅ 一致先右后左入栈
中序❌ 不一致指针一路向左,再处理
后序❌ 不一致反向收集 + 反转结果

结语:如果对你有帮助,请点赞,关注,收藏,你的支持就是我最大的鼓励!

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/5 4:20:21

Cincoze DX-1200工业边缘计算机深度评测与应用指南

1. Cincoze DX-1200无风扇嵌入式计算机概述Cincoze DX-1200是一款面向工业边缘计算和机器视觉应用的无风扇嵌入式计算机&#xff0c;搭载第12代Intel Core&#xff08;Alder Lake-S&#xff09;处理器&#xff0c;从赛扬到i9多种配置可选。这款设备采用坚固紧凑的铝制外壳设计&…

作者头像 李华
网站建设 2026/5/5 4:12:26

别再手算微带线宽了!用这个Matlab脚本,输入阻抗和板材参数直接出结果

微带线设计效率革命&#xff1a;Matlab自动化工具与工程实践指南 在射频电路设计领域&#xff0c;微带线作为最常见的传输线结构之一&#xff0c;其特性阻抗与物理尺寸的精确匹配直接决定了整个系统的性能表现。传统的手工计算或查表方法不仅耗时费力&#xff0c;而且在迭代设计…

作者头像 李华
网站建设 2026/5/5 4:11:43

3分钟掌握KMS_VL_ALL_AIO:Windows与Office智能激活的终极解决方案

3分钟掌握KMS_VL_ALL_AIO&#xff1a;Windows与Office智能激活的终极解决方案 【免费下载链接】KMS_VL_ALL_AIO Smart Activation Script 项目地址: https://gitcode.com/gh_mirrors/km/KMS_VL_ALL_AIO 还在为系统激活问题而烦恼吗&#xff1f;KMS_VL_ALL_AIO作为一款开…

作者头像 李华