当前位置: 首页 > news >正文

leetcode669.修剪二叉搜索树:递归法利用有序性精准剪枝

一、题目深度解析与BST特性应用

题目描述

给定一棵二叉搜索树(BST)和一个值区间[low, high],修剪BST使得所有节点的值都落在该区间内。修剪后的树必须保持BST的性质,且不能改变原有节点的相对位置关系

BST的核心特性应用

二叉搜索树的重要性质:

  • 左子树所有节点值 < 根节点值
  • 右子树所有节点值 > 根节点值
  • 中序遍历结果为严格递增序列

这些特性使得我们可以通过比较节点值与区间边界的大小关系,高效决定保留或舍弃哪些子树,从而实现精准剪枝。

二、递归解法的核心实现与逻辑框架

完整递归代码实现

/*** Definition for a binary tree node.* public class TreeNode {*     int val;*     TreeNode left;*     TreeNode right;*     TreeNode() {}*     TreeNode(int val) { this.val = val; }*     TreeNode(int val, TreeNode left, TreeNode right) {*         this.val = val;*         this.left = left;*         this.right = right;*     }* }*/
class Solution {public TreeNode trimBST(TreeNode root, int low, int high) {if(root == null){  // 终止条件:空节点直接返回return null;}// 情况1:当前节点值大于high,整棵右子树及当前节点都应舍弃if(root.val > high){return trimBST(root.left, low, high);  // 仅保留左子树修剪结果}// 情况2:当前节点值小于low,整棵左子树及当前节点都应舍弃if(root.val < low){return trimBST(root.right, low, high);  // 仅保留右子树修剪结果}// 情况3:当前节点值在区间内,递归修剪左右子树root.left = trimBST(root.left, low, high);root.right = trimBST(root.right, low, high);return root;  // 返回修剪后的当前节点}
}

核心设计解析:

  1. 递归终止条件

    • root为空,直接返回null
  2. 三种剪枝情况

    • 情况1:root.val > high
      当前节点及其右子树所有节点值均大于high,直接舍弃,递归处理左子树
    • 情况2:root.val < low
      当前节点及其左子树所有节点值均小于low,直接舍弃,递归处理右子树
    • 情况3:root.val ∈ [low, high]
      当前节点保留,递归修剪左右子树
  3. 递归路径选择

    • 根据当前节点值与区间边界的关系,选择性递归左子树或右子树
    • 确保每次递归处理的子树中可能存在符合条件的节点

三、核心问题解析:递归逻辑与剪枝策略

1. BST特性如何指导剪枝决策

关键性质推导:
  • 若root.val > high,则其右子树所有节点值均 > high → 整棵右子树可舍弃
  • 若root.val < low,则其左子树所有节点值均 < low → 整棵左子树可舍弃
  • 若root.val ∈ [low, high],则当前节点必须保留,但其左右子树可能仍需修剪
数学表达式:

设当前节点为root,则:

  • root.val > high时,修剪结果 = trimBST(root.left, low, high)
  • root.val < low时,修剪结果 = trimBST(root.right, low, high)
  • 否则,修剪结果 = 当前节点 + 修剪后的左右子树

2. 递归逻辑的切入点

递归步骤拆解:
  1. 终止条件检查:当前节点为空时返回null
  2. 值比较决策
    • 若当前节点值超出区间,直接舍弃并递归处理另一侧子树
    • 若当前节点值在区间内,递归修剪左右子树
  3. 子树重构:将修剪后的左右子树连接到当前节点
  4. 返回结果:向上层返回修剪后的当前子树
关键代码逻辑:
// 舍弃右子树,递归处理左子树
if(root.val > high) return trimBST(root.left, low, high);// 舍弃左子树,递归处理右子树
if(root.val < low) return trimBST(root.right, low, high);// 保留当前节点,递归修剪左右子树
root.left = trimBST(root.left, low, high);
root.right = trimBST(root.right, low, high);

四、递归流程深度模拟:以示例BST为例

示例BST结构与修剪区间:

        3/ \0   4\2/1

修剪区间[1, 3]

递归调用过程:

  1. 根节点3的递归

    • 比较:3 ∈ [1, 3] → 保留当前节点
    • 递归左子树0
    • 递归右子树4
  2. 左子树0的递归

    • 比较:0 < 1 → 舍弃当前节点及其左子树
    • 递归右子树2
  3. 节点2的递归

    • 比较:2 ∈ [1, 3] → 保留当前节点
    • 递归左子树1
    • 递归右子树null
  4. 节点1的递归

    • 比较:1 ∈ [1, 3] → 保留当前节点
    • 左右子树均为null → 返回节点1
  5. 右子树4的递归

    • 比较:4 > 3 → 舍弃当前节点及其右子树
    • 递归左子树null → 返回null

修剪后的BST结构:

        3/2/1

五、算法复杂度分析

1. 时间复杂度

  • O(n):每个节点最多访问一次,n为树的节点数
  • 递归过程中每个节点执行常数级判断,总时间线性于节点数

2. 空间复杂度

  • O(h):h为树的高度(递归栈深度)
    • 平衡树:h=logn,空间复杂度O(logn)
    • 最坏情况(链表树):h=n,空间复杂度O(n)

3. 与迭代解法对比

方法时间复杂度空间复杂度代码简洁度
递归法O(n)O(h)
迭代法O(n)O(h)

六、核心技术点总结:递归剪枝的三大关键

1. BST有序性的充分利用

  • 路径压缩:根据节点值与区间的关系,每次递归排除一半子树
  • 剪枝策略:直接舍弃不符合条件的整棵子树,无需遍历其中节点

2. 递归设计的精妙之处

  • 自底向上重构:递归返回值自然实现子树重构
  • 简洁逻辑:通过三种情况处理所有可能的节点值分布

3. 区间边界的数学意义

  • 闭区间保证[low, high]确保包含边界值的节点被保留
  • 传递不变性:递归过程中始终保持区间参数不变,确保剪枝一致性

七、常见误区与优化建议

1. 错误的剪枝判断

  • 误区:使用root.val >= highroot.val <= low作为剪枝条件
    // 错误示例:可能误删边界值节点
    if(root.val >= high) return trimBST(root.left, low, high);
    
  • 正确逻辑:严格使用><,确保边界值节点被保留

2. 不必要的子树遍历

  • 优化点:当节点值超出区间时,直接舍弃整棵子树
  • 错误做法:递归遍历超出区间的子树后再判断
    // 低效做法:浪费时间遍历无效子树
    TreeNode left = trimBST(root.left, low, high);
    if(root.val > high) return left;
    

3. 迭代实现的可行性

// 迭代法框架(代码不完整,仅作示意)
Stack<TreeNode> stack = new Stack<>();
stack.push(root);
while(!stack.isEmpty()) {TreeNode node = stack.pop();if(node.val > high) {// 处理左子树} else if(node.val < low) {// 处理右子树} else {// 处理左右子树}
}
  • 优势:避免递归栈溢出风险
  • 劣势:代码复杂度显著增加,需手动维护栈结构

八、总结:递归剪枝是BST特性的最佳实践

本算法通过递归方式,充分利用BST的有序性,将修剪操作转化为高效的子树排除问题。核心在于:

  1. 数值比较指导剪枝决策:根据节点值与区间边界的关系,直接舍弃无效子树
  2. 三种剪枝场景处理:覆盖所有可能的节点值分布情况
  3. 递归实现子树重构:通过递归返回值自然实现树结构的调整

理解这种递归解法的关键在于把握BST的数值分布特性,将修剪问题转化为区间定位问题。在实际工程中,这种利用数据结构固有特性的优化思路,是提升算法效率的重要方向。

相关文章:

  • OAuth详解和应用
  • Netty 实战篇:为 Netty RPC 框架增加超时控制与重试机制,防止系统雪崩
  • 实验设计与分析(第6版,Montgomery)第5章析因设计引导5.7节思考题5.11 R语言解题
  • Ubuntu 安装 FSL 及多模态脑MRI的去颅骨处理(含 HD-BET 深度学习方法)
  • C#中数据绑定的简单例子
  • 深入浅出:使用DeepSeek开发小程序的完整指南
  • vb.net oledb-Access 数据库本身不支持命名参数,赋值必须和参数顺序一致才行
  • STM32 搭配 嵌入式SD卡在智能皮电手环中的应用全景评测
  • 引领机器人交互未来!MANUS数据手套解锁精准手部追踪
  • 改写自己的浏览器插件工具 myChromeTools
  • 4.2.3 Spark SQL 手动指定数据源
  • 手撕Java+硅基流动实现MCP服务器教程
  • Fusion引擎赋能:流利说如何用阿里云Serverless Spark实现数仓计算加速
  • 深入 RAG(检索增强生成)系统架构:如何构建一个能查资料的大语言模型系统
  • 学习路之PHP--easyswoole_panel安装使用
  • Css样式中设置gap: 12px以后左右出现距离问题解析
  • 半导体晶圆制造洁净厂房的微振控制方案-江苏泊苏系统集成有限公司
  • 深度学习笔记25-RNN心脏病预测(Pytorch)
  • Haption在危险、挑战性或受限环境中操作的情况提供了一种创新的遥操作解决方案
  • 雷达目标起伏特性简析
  • 做本地网站怎么挣钱/贴吧aso优化贴吧
  • jfinal怎么做网站/b站推广入口2023年
  • 舆情网站大全/网络优化工程师是干什么的
  • 十堰网站建设专家/外贸网站制作公司
  • 传统企业建设营销型网站/热点新闻事件及观点
  • wordpress做教育网站/百度一下手机版首页