LeetCode Hot100(27/100)——94. 二叉树的中序遍历

文章目录

一、题目理解

题目链接:https://leetcode.cn/problems/binary-tree-inorder-traversal

题目描述:

给定一个二叉树的根节点 root ,返回它的 中序遍历(Left → Root → Right)。

示例:

输入:

复制代码
    1
     \
      2
     /
    3

输出:

复制代码
[1, 3, 2]

二、中序遍历的核心概念

中序遍历(Inorder Traversal) 顺序规则如下:

  1. 访问左子树;
  2. 访问当前节点;
  3. 访问右子树。

即:Left → Root → Right


思维导图

中序遍历
递归解法
访问左子树
访问根节点
访问右子树
迭代解法(栈)
使用栈模拟递归
不断向左入栈
弹出节点访问
转向右子树
Morris遍历(进阶)
不使用栈和递归
改变指针结构
空间O(1)


三、解法一:递归法(最容易理解)

思路:

递归本质是调用栈来管理访问顺序。每到一个节点,先递归访问它的左子树,再处理当前节点,最后访问它的右子树。


递归流程图



开始
当前节点为空?
返回
递归遍历左子树
访问当前节点,加入结果集
递归遍历右子树
结束


Java代码实现

java 复制代码
import java.util.*;

public class Solution {
    public List<Integer> inorderTraversal(TreeNode root) {
        List<Integer> res = new ArrayList<>();
        dfs(root, res);
        return res;
    }

    private void dfs(TreeNode node, List<Integer> res) {
        if (node == null) return;
        dfs(node.left, res);
        res.add(node.val);
        dfs(node.right, res);
    }
}

复杂度分析

项目 分析
时间复杂度 O(n),每个节点被访问一次
空间复杂度 O(n),主要来自递归栈的深度(最坏为O(n))

四、解法二:迭代法(使用栈模拟递归)

思路:

  • 使用一个栈手动控制遍历顺序;
  • 不断向左走并压栈;
  • 当左边到底后,弹出节点访问,再转向右子树;
  • 重复这个过程直到栈为空且当前节点为空。

栈模拟流程图





初始化栈,当前节点为root
当前节点不为空或栈不为空?
结束
当前节点不为空?
将当前节点入栈,移动到左节点
弹出栈顶节点并访问
移动到右节点


Java实现代码

java 复制代码
import java.util.*;

public class Solution {
    public List<Integer> inorderTraversal(TreeNode root) {
        List<Integer> res = new ArrayList<>();
        Deque<TreeNode> stack = new LinkedList<>();
        TreeNode curr = root;

        while (curr != null || !stack.isEmpty()) {
            while (curr != null) {
                stack.push(curr);
                curr = curr.left;
            }
            curr = stack.pop();
            res.add(curr.val);
            curr = curr.right;
        }

        return res;
    }
}

复杂度分析

项目 分析
时间复杂度 O(n)
空间复杂度 O(n)(显式栈存储节点)

五、解法三(进阶):Morris遍历(O(1)空间)

原理:

Morris遍历通过在树中建立临时线索(Thread),在不使用栈或递归的情况下也能实现中序遍历。

核心思想:

  1. 如果当前节点没有左子树,访问它,然后走右子树。
  2. 如果有左子树,就找到左子树最右节点(前驱节点):
    • 若其右指针为空,则将其右指针指向当前节点(建立线索),然后进入左子树;
    • 若其右指针指向当前节点(说明左子树已访问完),则将其恢复空,访问当前节点,进入右子树。

Morris遍历时序图

前驱节点 左子树 当前节点 前驱节点 左子树 当前节点 寻找左子树的最右节点 若左子树不为空 找到左子树最右节点 建立右指针线索 左子树遍历 恢复右指针为空 访问节点并转向右子树


Java实现代码

java 复制代码
import java.util.*;

public class Solution {
    public List<Integer> inorderTraversal(TreeNode root) {
        List<Integer> res = new ArrayList<>();
        TreeNode curr = root;

        while (curr != null) {
            if (curr.left == null) {
                res.add(curr.val);
                curr = curr.right;
            } else {
                TreeNode pre = curr.left;
                while (pre.right != null && pre.right != curr) {
                    pre = pre.right;
                }
                if (pre.right == null) {
                    pre.right = curr;
                    curr = curr.left;
                } else {
                    pre.right = null;
                    res.add(curr.val);
                    curr = curr.right;
                }
            }
        }
        return res;
    }
}

复杂度分析

项目 分析
时间复杂度 O(n),每个节点访问两次
空间复杂度 O(1),未使用递归或栈
相关推荐
九.九13 小时前
CANN HCOMM 底层机制深度解析:集合通信算法实现、RoCE 网络协议栈优化与多级同步原语
网络·网络协议·算法
C++ 老炮儿的技术栈13 小时前
Qt Creator中不写代如何设置 QLabel的颜色
c语言·开发语言·c++·qt·算法
子春一13 小时前
Flutter for OpenHarmony:构建一个 Flutter 数字消消乐游戏,深入解析网格状态管理、合并算法与重力系统
算法·flutter·游戏
草履虫建模19 小时前
力扣算法 1768. 交替合并字符串
java·开发语言·算法·leetcode·职场和发展·idea·基础
naruto_lnq21 小时前
分布式系统安全通信
开发语言·c++·算法
Jasmine_llq1 天前
《P3157 [CQOI2011] 动态逆序对》
算法·cdq 分治·动态问题静态化+双向偏序统计·树状数组(高效统计元素大小关系·排序算法(预处理偏序和时间戳)·前缀和(合并单个贡献为总逆序对·动态问题静态化
爱吃rabbit的mq1 天前
第09章:随机森林:集成学习的威力
算法·随机森林·集成学习
(❁´◡`❁)Jimmy(❁´◡`❁)1 天前
Exgcd 学习笔记
笔记·学习·算法
YYuCChi1 天前
代码随想录算法训练营第三十七天 | 52.携带研究材料(卡码网)、518.零钱兑换||、377.组合总和IV、57.爬楼梯(卡码网)
算法·动态规划