LeetCode-hot100——验证二叉搜索树

题目:验证二叉搜索树

问题定义

验证二叉搜索树(BST)是判断一棵棵二叉树是否满足以下特性:

  • 左子树中所有节点的值都严格小于当前节点的值
  • 右子树中所有节点的值都严格大于当前节点的值
  • 左、右子树也必须是有效的二叉搜索树

方法思路

采用「递归 + 范围约束」策略,核心思想是为每个节点维护一个合法的取值范围(最小值min和最大值max):

  1. 根节点的初始范围为(-∞, +∞)(用 LONG_MIN LONG_MAX 表示)
  2. 左子树的取值范围为(父节点的min, 父节点的值)
  3. 右子树的取值范围为(父节点的值, 父节点的max)
  4. 若任何节点的值超出其合法范围,则该树不是有效的 BST

代码实现

cpp 复制代码
class Solution {
public:
    // 辅助递归函数:验证以root为根的子树是否为BST
    // min: 当前子树节点允许的最小值(开区间)
    // max: 当前子树节点允许的最大值(开区间)
    bool BST(TreeNode* root, long long min, long long max) {
        // 空节点视为有效BST
        if (root == nullptr) return true;
        
        // 若当前节点值超出范围,则不是BST
        if (root->val >= max || root->val <= min) return false;
        
        // 递归验证左子树和右子树
        // 左子树最大值为当前节点值,右子树最小值为当前节点值
        return BST(root->left, min, root->val) && BST(root->right, root->val, max);
    } 
    
    // 主函数:验证整棵树是否为BST
    bool isValidBST(TreeNode* root) {
        // 初始范围设为long long的极值,避免int边界值问题
        return BST(root, LONG_MIN, LONG_MAX);
    }
};

样例演示

以示例[6,3,8,2,5,7,9]为例;

cpp 复制代码
          6
        /   \
       3     8
      / \   / \
     2   5 7   9

根节点验证

cpp 复制代码
调用  BST(6, LONG_MIN, LONG_MAX)
  • 节点值:6
  • 合法范围:(-∞, +∞)
  • 验证:6 在范围内
  • 下一步:验证左子树 (3) 和右子树 (8)

左子树验证(节点 3)

cpp 复制代码
调用  BST(3, LONG_MIN, 6)
  • 节点值:3
  • 合法范围:(-∞, 6)(因为父节点是 6,左子树最大值为 6)
  • 验证:3 在范围内
  • 下一步:验证左子树 (2) 和右子树 (5)

左子树的左子树(节点 2)

cpp 复制代码
调用  BST(2, LONG_MIN, 3)
  • 节点值:2
  • 合法范围:(-∞, 3)(父节点是 3,左子树最大值为 3)
  • 验证:2 在范围内
  • 节点 2 的左右子树均为空,返回 true

左子树的右子树(节点 5)

cpp 复制代码
调用  BST(5, 3, 6)
  • 节点值:5
  • 合法范围:(3, 6) (父节点是 3,右子树最小值为 3;祖父节点是 6,左子树最大值为 6)
  • 验证:5 在范围内 ✅
  • 节点 5 的左右子树均为空,返回 true
  • 左子树 (3) 的验证结果:true && true = true

右子树验证(节点 8)

cpp 复制代码
调用  BST(8, 6, LONG_MAX)
  • 节点值:8
  • 合法范围:(6, +∞)(父节点是 6,右子树最小值为 6)
  • 验证:8 在范围内 ✅
  • 下一步:验证左子树 (7) 和右子树 (9)

右子树的左子树(节点 7)

cpp 复制代码
调用 BST(7, 6, 8)
  • 节点值:7
  • 合法范围:(6, 8)(父节点是 8,左子树最大值为 8;祖父节点是 6,右子树最小值为 6)
  • 验证:7 在范围内 ✅
  • 节点 7 的左右子树均为空,返回 true

右子树的右子树(节点 9)

cpp 复制代码
调用  BST(9, 8, LONG_MAX)
  • 节点值:9
  • 合法范围:(8, +∞)(父节点是 8,右子树最小值为 8)
  • 验证:9 在范围内 ✅
  • 节点 9 的左右子树均为空,返回 true
  • 右子树 (8) 的验证结果:true && true = true

验证结果

根节点的验证结果:左子树结果 && 右子树结果 = true && true = true 因此该树是有效的二叉搜索树

关键细节解析

  1. 范围约束机制:每个节点的取值范围由整条祖先路径共同决定,例如节点 5 的范围 (3,6) 同时受父节点 3 和祖父节点 6 的约束。

  2. 数据类型选择

    • 使用long long而非int作为范围边界类型;
    • 避免当节点值为 INT_MAX INT_MIN 时出现的溢出问题(如节点值为INT_MAX时,其父节点的 max 若为INT_MAX会导致判断错误);
  3. 严格不等式

    • 条件root->val >= max || root->val <= min确保了 BST 中不允许存在相等的值;
    • 符合二叉搜索树 "严格小于 / 大于" 的定义;
相关推荐
2301_81123298几秒前
低延迟系统C++优化
开发语言·c++·算法
alphaTao2 分钟前
LeetCode 每日一题 2026/1/26-2026/2/1
算法·leetcode
我能坚持多久3 分钟前
D20—C语言文件操作详解:从基础到高级应用
c语言·开发语言
txinyu的博客18 分钟前
解析muduo源码之 ThreadLocal.h
c++
向哆哆19 分钟前
构建跨端健身俱乐部管理系统:Flutter × OpenHarmony 的数据结构与设计解析
数据结构·flutter·鸿蒙·openharmony·开源鸿蒙
橘子师兄20 分钟前
C++AI大模型接入SDK—ChatSDK封装
开发语言·c++·人工智能·后端
Christo323 分钟前
TFS-2026《Fuzzy Multi-Subspace Clustering 》
人工智能·算法·机器学习·数据挖掘
上天_去_做颗惺星 EVE_BLUE34 分钟前
Docker高效使用指南:从基础到实战模板
开发语言·ubuntu·docker·容器·mac·虚拟环境
2401_8576835435 分钟前
C++中的原型模式
开发语言·c++·算法
s1hiyu1 小时前
C++动态链接库开发
开发语言·c++·算法