【LeetCode 热题100】【二叉树构造题精讲:前序 + 中序建树 & 有序数组构造 BST】(详细解析)(Go语言版)

简介: 本文详细解析了二叉树构造的两类经典问题:通过前序与中序遍历重建二叉树(LeetCode 105),以及将有序数组转化为平衡二叉搜索树(BST,LeetCode 108)。文章从核心思路、递归解法到实现细节逐一拆解,强调通过索引控制子树范围以优化性能,并对比两题的不同构造逻辑。最后总结通用构造套路,提供进阶思考方向,帮助彻底掌握二叉树构造类题目。

🌱 二叉树构造题精讲:前序 + 中序建树 & 有序数组构造 BST

本文围绕二叉树的两类构造类题目展开解析:

    1. 从前序与中序遍历序列构造二叉树
    1. 将有序数组转换为二叉搜索树

我们将从「已知遍历构造树」和「平衡构造 BST」两个角度,拆解树结构的构建逻辑,彻底吃透构造题型。


📌 题目一:105. 从前序与中序遍历序列构造二叉树

📝 题目描述

给定两棵树的遍历序列:

  • preorder(前序遍历):根 -> 左 -> 右
  • inorder(中序遍历):左 -> 根 -> 右

请根据这两种遍历构造原始二叉树。


🔍 解题思路

核心思路:

  1. 前序遍历的第一个元素一定是根节点;
  2. 在中序遍历中找到这个根节点的位置,可以确定左右子树的元素范围;
  3. 递归构造左右子树。

✅ 解法:递归构造

func buildTree(preorder []int, inorder []int) *TreeNode {
   
    inMap := map[int]int{
   }
    for i, val := range inorder {
   
        inMap[val] = i
    }

    var build func(pl, pr, il, ir int) *TreeNode
    build = func(pl, pr, il, ir int) *TreeNode {
   
        if pl > pr || il > ir {
   
            return nil
        }

        rootVal := preorder[pl]
        root := &TreeNode{
   Val: rootVal}
        idx := inMap[rootVal]
        leftSize := idx - il

        root.Left = build(pl+1, pl+leftSize, il, idx-1)
        root.Right = build(pl+leftSize+1, pr, idx+1, ir)

        return root
    }

    return build(0, len(preorder)-1, 0, len(inorder)-1)
}

📘 思路详解

  • 使用 inMap 来快速定位中序中某个值的位置,避免每次线性搜索;
  • 用索引控制遍历范围,不要切片传参,会影响性能
  • 每次递归缩小当前处理的 preorder 和 inorder 区间。

⚠️ 注意事项:

  • preorder[pl] 是当前子树的根节点;
  • 左子树的大小为 idx - il
  • 左右子树递归时注意索引边界不要写错。

📌 题目二:108. 将有序数组转换为二叉搜索树

📝 题目描述

给你一个升序排序的整数数组,请你将其转化为一棵高度平衡的二叉搜索树(BST)


🔍 解题思路

关键点:

  • 数组有序 → 可用中间元素构建根节点;
  • 左边递归为左子树,右边递归为右子树;
  • 中间元素选择策略:可以取中间偏左或偏右均可。

✅ 解法:递归 + 中点分割

func sortedArrayToBST(nums []int) *TreeNode {
   
    var build func(left, right int) *TreeNode
    build = func(left, right int) *TreeNode {
   
        if left > right {
   
            return nil
        }
        mid := (left + right) / 2
        root := &TreeNode{
   Val: nums[mid]}
        root.Left = build(left, mid-1)
        root.Right = build(mid+1, right)
        return root
    }

    return build(0, len(nums)-1)
}

💭 思维补充

  • 二叉搜索树要求:左 < 根 < 右;
  • 高度平衡树要求:每个节点左右子树高度差不超过 1;
  • 因此「中间作为根」是构造平衡 BST 的最优策略。

🧠 总结 & 对比

题目 类型 输入 输出 核心操作
105 构造普通二叉树 前序 + 中序遍历 树结构 递归 + 分治(索引控制)
108 构造平衡 BST 有序数组 BST 树结构 递归 + 二分中点

🎯 通用构造套路小结:

  1. 明确根节点从何而来(前序 or 中点);
  2. 找到左右子树的边界
  3. 索引控制子问题的范围
  4. 构建节点,递归处理左右子树;
  5. 特别注意边界条件与 base case

✨ 进阶思考

  • 如果输入是 中序 + 后序,你还能反推出树吗?
  • 如果输入是 BST + 任意遍历,你能判断树结构吗?

这些问题都是构造类题目的常见变体,建议从这两题出发逐步拓展思维路径。


下一篇将带你探索搜索树相关的问题,从验证 BST 到查找第 K 小元素,一起掌握搜索树的价值!

如有帮助,欢迎点赞收藏,更多结构题内容持续更新中 🧠💡


目录
相关文章
|
9月前
|
数据采集 数据挖掘 测试技术
Go与Python爬虫实战对比:从开发效率到性能瓶颈的深度解析
本文对比了Python与Go在爬虫开发中的特点。Python凭借Scrapy等框架在开发效率和易用性上占优,适合快速开发与中小型项目;而Go凭借高并发和高性能优势,适用于大规模、长期运行的爬虫服务。文章通过代码示例和性能测试,分析了两者在并发能力、错误处理、部署维护等方面的差异,并探讨了未来融合发展的趋势。
918 0
|
7月前
|
Cloud Native 安全 Java
Go语言深度解析:从入门到精通的完整指南
🌟蒋星熠Jaxonic,Go语言探索者。深耕云计算、微服务与并发编程,以代码为笔,在二进制星河中书写极客诗篇。分享Go核心原理、性能优化与实战架构,助力开发者掌握云原生时代利器。#Go语言 #并发编程 #性能优化
625 43
Go语言深度解析:从入门到精通的完整指南
|
8月前
|
Cloud Native 安全 Java
Go语言深度解析:从入门到精通的完整指南
🌟 蒋星熠Jaxonic,执着的星际旅人,用Go语言编写代码诗篇。🚀 Go语言以简洁、高效、并发为核心,助力云计算与微服务革新。📚 本文详解Go语法、并发模型、性能优化与实战案例,助你掌握现代编程精髓。🌌 从goroutine到channel,从内存优化到高并发架构,全面解析Go的强大力量。🔧 实战构建高性能Web服务,展现Go在云原生时代的无限可能。✨ 附技术对比、最佳实践与生态全景,带你踏上Go语言的星辰征途。#Go语言 #并发编程 #云原生 #性能优化
|
7月前
|
存储 人工智能 算法
从零掌握贪心算法Java版:LeetCode 10题实战解析(上)
在算法世界里,有一种思想如同生活中的"见好就收"——每次做出当前看来最优的选择,寄希望于通过局部最优达成全局最优。这种思想就是贪心算法,它以其简洁高效的特点,成为解决最优问题的利器。今天我们就来系统学习贪心算法的核心思想,并通过10道LeetCode经典题目实战演练,带你掌握这种"步步为营"的解题思维。
|
11月前
|
存储 设计模式 安全
Go 语言单例模式全解析:从青铜到王者段位的实现方案
单例模式确保一个类只有一个实例,并提供全局访问点,适用于日志、配置管理、数据库连接池等场景。在 Go 中,常用实现方式包括懒汉模式、饿汉模式、双重检查锁定,最佳实践是使用 `sync.Once`,它并发安全、简洁高效。本文详解各种实现方式的优缺点,并提供代码示例与最佳应用建议。
354 5
|
9月前
|
缓存 监控 安全
告别缓存击穿!Go 语言中的防并发神器:singleflight 包深度解析
在高并发场景中,多个请求同时访问同一资源易导致缓存击穿、数据库压力过大。Go 语言提供的 `singleflight` 包可将相同 key 的请求合并,仅执行一次实际操作,其余请求共享结果,有效降低系统负载。本文详解其原理、实现及典型应用场景,并附示例代码,助你掌握高并发优化技巧。
646 0
|
9月前
|
数据采集 JSON Go
Go语言实战案例:实现HTTP客户端请求并解析响应
本文是 Go 网络与并发实战系列的第 2 篇,详细介绍如何使用 Go 构建 HTTP 客户端,涵盖请求发送、响应解析、错误处理、Header 与 Body 提取等流程,并通过实战代码演示如何并发请求多个 URL,适合希望掌握 Go 网络编程基础的开发者。
|
Unix Shell Linux
LeetCode刷题 Shell编程四则 | 194. 转置文件 192. 统计词频 193. 有效电话号码 195. 第十行
本文提供了几个Linux shell脚本编程问题的解决方案,包括转置文件内容、统计词频、验证有效电话号码和提取文件的第十行,每个问题都给出了至少一种实现方法。
410 6
LeetCode刷题 Shell编程四则 | 194. 转置文件 192. 统计词频 193. 有效电话号码 195. 第十行
|
Python
【Leetcode刷题Python】剑指 Offer 32 - III. 从上到下打印二叉树 III
本文介绍了两种Python实现方法,用于按照之字形顺序打印二叉树的层次遍历结果,实现了在奇数层正序、偶数层反序打印节点的功能。
222 6
|
搜索推荐 索引 Python
【Leetcode刷题Python】牛客. 数组中未出现的最小正整数
本文介绍了牛客网题目"数组中未出现的最小正整数"的解法,提供了一种满足O(n)时间复杂度和O(1)空间复杂度要求的原地排序算法,并给出了Python实现代码。
497 2

热门文章

最新文章