105. 从前序与中序遍历序列构造二叉树

105. 从前序与中序遍历序列构造二叉树

中等

给定两个整数数组 preorderinorder ,其中 preorder 是二叉树的先序遍历inorder 是同一棵树的中序遍历,请构造二叉树并返回其根节点。

示例 1:

复制代码
输入: preorder = [3,9,20,15,7], inorder = [9,3,15,20,7]
输出: [3,9,20,null,null,15,7]

示例 2:

复制代码
输入: preorder = [-1], inorder = [-1]
输出: [-1]

提示:

  • 1 <= preorder.length <= 3000
  • inorder.length == preorder.length
  • -3000 <= preorder[i], inorder[i] <= 3000
  • preorderinorder无重复 元素
  • inorder 均出现在 preorder
  • preorder 保证 为二叉树的前序遍历序列
  • inorder 保证 为二叉树的中序遍历序列

📝 核心笔记:从前序与中序遍历构造二叉树 (Construct Tree from Preorder & Inorder)

1. 核心思想 (一句话总结)

"前序定根,中序分左右。"

  • 前序遍历 ( Root -> Left -> Right**)** :告诉我们谁是根节点(永远是第一个元素)。
  • 中序遍历 ( Left -> Root -> Right**)** :告诉我们左右子树有多大(找到根节点的位置,左边就是左子树,右边就是右子树)。
  • 利用这个性质,我们可以把数组"切"成两半,递归地去构建左子树和右子树。
2. 算法流程 (分治切割)
  1. 找根 (Locate Root)
    • rootVal = preorder[0]
  1. 算大小 (Calculate Size)
    • inorder 数组里找到 rootVal 的下标 i
    • leftSize = i - 0(中序数组中,根节点左边的元素个数)。
  1. 切数组 (Slice & Recurse)
    • 左子树
      • 前序:[1, 1 + leftSize]
      • 中序:[0, leftSize]
    • 右子树
      • 前序:[1 + leftSize, end]
      • 中序:[leftSize + 1, end]
  1. 连接 (Link)root.left = recurse(...), root.right = recurse(...)
3. ⚠️ 性能优化的进阶 (面试必考)

您的写法虽然逻辑完美,但存在两个性能瓶颈:

  1. 时间indexOf 每次都是 O(N) 扫描,总复杂度退化为 O(N\^2)
  2. 空间Arrays.copyOfRange 每次都开辟新数组,空间消耗巨大。

优化方案

  • 哈希表优化 :一开始把 inorder 的元素和下标存入 HashMap,实现 找根。
  • 指针优化 :不切数组,而是传递 下标范围 (preStart, preEnd, inStart, inEnd)。
🔍 代码回忆清单
复制代码
// 题目:LC 105. Construct Binary Tree from Preorder and Inorder Traversal
class Solution {
    // 缓存中序遍历的 (值 -> 下标),加速定位
    private Map<Integer, Integer> indexMap;

    public TreeNode buildTree(int[] preorder, int[] inorder) {
        indexMap = new HashMap<>();
        for (int i = 0; i < inorder.length; i++) {
            indexMap.put(inorder[i], i);
        }
        // 传递数组范围,而不是拷贝数组
        return build(preorder, 0, preorder.length - 1, 
                     inorder, 0, inorder.length - 1);
    }

    private TreeNode build(int[] preorder, int preStart, int preEnd, 
                           int[] inorder, int inStart, int inEnd) {
        // 1. Base Case: 范围无效 (左指针跑过右指针)
        if (preStart > preEnd) {
            return null;
        }

        // 2. 找根节点值
        int rootVal = preorder[preStart];
        TreeNode root = new TreeNode(rootVal);

        // 3. 找根在中序中的位置 (O(1) 查表)
        int inRootIndex = indexMap.get(rootVal);
        
        // 4. 计算左子树的节点数量
        int leftSize = inRootIndex - inStart;

        // 5. 递归构建 (最容易写错下标的地方,建议画图)
        
        // 左子树:
        // Pre: [preStart + 1, preStart + leftSize] (根后面紧接着就是左子树)
        // In:  [inStart, inRootIndex - 1] (根左边全是左子树)
        root.left = build(preorder, preStart + 1, preStart + leftSize, 
                          inorder, inStart, inRootIndex - 1);

        // 右子树:
        // Pre: [preStart + leftSize + 1, preEnd] (剩下的部分)
        // In:  [inRootIndex + 1, inEnd] (根右边全是右子树)
        root.right = build(preorder, preStart + leftSize + 1, preEnd, 
                           inorder, inRootIndex + 1, inEnd);

        return root;
    }
}
⚡ 快速复习 CheckList (易错点)
  • \] **为什么前序数组也要维护** **start****和** **end****?**

    • 虽然我们只用 preorder[preStart] 找根,但在递归右子树时,我们需要计算右子树在前序数组中的起始位置(也就是跳过左子树那一段)。
  • \] **下标偏移量怎么算?**

    • 左子树的前序结束位置preStart + leftSize
    • 右子树的前序起始位置preStart + leftSize + 1
    • 这个公式一定要理解记忆,死记硬背容易忘。
  • \] **如果有重复元素怎么办?**

    • 题目通常保证树中没有重复元素。如果有,结果不唯一,或者需要更多信息。
🖼️ 数字演练

Pre: [3, 9, 20, 15, 7]

In: [9, 3, 15, 20, 7]

  1. Level 1:
    • Root = 3 (pre[0]).
    • Inorder Index of 3 = 1.
    • Left Size = 1 - 0 = 1.
    • Left Recurse : Pre [9], In [9]. -> Returns Node(9).
    • Right Recurse : Pre [20, 15, 7], In [15, 20, 7].
  1. Level 2 (Right):
    • Root = 20.
    • Inorder Index of 20 = 3.
    • Left Size (of 20) = 3 - 2 = 1 (元素 15).
    • Left Recurse : Pre [15], In [15]. -> Returns Node(15).
    • Right Recurse : Pre [7], In [7]. -> Returns Node(7).
相关推荐
样例过了就是过了1 小时前
LeetCodere热题100 最小覆盖子串
数据结构·算法·leetcode
前路不黑暗@1 小时前
Java项目:Java脚手架项目的地图的POJO
android·java·开发语言·spring boot·学习·spring cloud·maven
2501_926978331 小时前
分形时空理论框架:从破缺悖论到意识宇宙的物理学新范式引言(理论概念版)--AGI理论系统基础1.1
java·服务器·前端·人工智能·经验分享·agi
西门吹雪分身1 小时前
K8S之Pod调度
java·容器·kubernetes·k8s
弹简特1 小时前
【JavaEE08-后端部分】SpringMVC03-SpringMVC第二大核心处理请求之Cookie/Session和获取header
java·spring boot·spring·java-ee
追随者永远是胜利者1 小时前
(LeetCode-Hot100)10. 正则表达式匹配
java·算法·leetcode·go
We་ct2 小时前
LeetCode 146. LRU缓存:题解+代码详解
前端·算法·leetcode·链表·缓存·typescript
烟花落o2 小时前
【数据结构系列03】链表的回文解构、相交链表
数据结构·算法·链表·刷题
fu的博客2 小时前
【数据结构4】单向循环链表实现
数据结构·链表