98. 验证二叉搜索树

一、算法逻辑(逐步通顺讲解每一步思路)

这段算法使用了一种递归的思路:
每个节点返回它所在子树的 最小值和最大值,并在返回的过程中检查 BST 的合法性。

✅ 1️⃣ 定义递归函数 dfs(node),其含义是:
对于以 node 为根的子树,返回其值域范围 (最小值, 最大值)

✅ 2️⃣ 处理空节点(递归边界):
node is None 时,说明是空子树,返回 (inf, -inf)

  • 空树最大值是 -inf,最小值是 +inf,便于后续比较中不干扰当前节点的合法判断。

✅ 3️⃣ 分别递归左右子树:

获取当前节点左右子树的最大/最小值。

✅ 4️⃣ 当前节点合法性判断(BST 核心条件):

  • 当前节点的值 x = node.val,必须满足:

即:当前节点必须大于左子树的最大值,小于右子树的最小值。

如果不满足,则立即返回非法标记 (−inf, +inf),用来“污染”父节点的判断。

✅ 5️⃣ 当前子树的值域汇总:
如果合法,则当前子树的最小值为 min(l_min, x),最大值为 max(r_max, x),向上传递。

✅ 6️⃣ 最终检查:
顶层返回的是根节点子树的值域,如果合法,就不等于 (-inf, +inf);否则被污染为非法。


二、核心点总结

该算法的核心是:

后序遍历地收集每棵子树的最值信息,同时判断当前节点是否满足 BST 性质。

✅ 通过最小/最大值来传递合法性,思路独特;
✅ 利用 (−inf, +inf) 作为非法标志,避免使用全局变量;
✅ 后序遍历先处理左右子树,再判断当前节点,能做到“非法剪枝”;
✅ 可扩展性强:该模式也可以改写为同时处理 AVL 树、红黑树的平衡性检测等。

class Solution:def isValidBST(self, root: Optional[TreeNode]) -> bool:def dfs(node: Optional[TreeNode]) -> Tuple:if node is None:return inf, -infl_min, l_max = dfs(node.left)r_min, r_max = dfs(node.right)x = node.val# 也可以在递归完左子树之后立刻判断,如果发现不是二叉搜索树,就不用递归右子树了if x <= l_max or x >= r_min:return -inf, infreturn min(l_min, x), max(r_max, x)return dfs(root)[1] != inf

三、时间复杂度分析

  • 每个节点都被访问一次;

  • 每次访问包含常数级操作(比较、返回值)。

所以时间复杂度为:

O(n),其中 n 是树的节点数


四、空间复杂度分析

  • 主要是递归调用栈;

  • 最坏情况下是链状结构,深度为 n

  • 最好是平衡树,深度为 log n

所以空间复杂度为:

O(h),其中 h 是树的高度,最坏为 O(n),平均为 O(log n)


✅ 总结一句话

本算法通过后序遍历传递 (最小值, 最大值) 并在每一层校验 BST 条件,以递归+返回值方式 elegantly 判断整棵树是否为 BST。其巧妙之处在于使用返回值同时传递判断信息与范围信息,无状态、无额外全局变量、思路干净优雅。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/web/87655.shtml
繁体地址,请注明出处:http://hk.pswp.cn/web/87655.shtml
英文地址,请注明出处:http://en.pswp.cn/web/87655.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Flink-Source算子点位提交问题(Earliest)

背景 最近在做 Flink 任务数据源切换时遇到 offset 消费问题&#xff0c;遂写篇文章记录下来。 切换时只修改了 source 算子的 topic&#xff0c;uid 等其他信息保持不变&#xff1a; 发布时&#xff0c;发现算子的消费者点位重置为earliest&#xff0c;导致消息积压。消息积…

如何录制带备注的演示文稿(LaTex Beamer + Pympress)

参考文献&#xff1a; Pympress 官网Avidemux 官网Audacity 官网FFmpeg 官网2025年度25大视频剪辑软件推荐2025最新音频降噪软件盘点&#xff0c;从入门到专业的6个高效工具如何用一段音频替换mp4视频格式的原有音频&#xff1f;免费简单易用的视频剪切编辑工具—AvidemuxFFmp…

VS Code 的 Copilot Chat 扩展程序

安装与启用 Copilot Chat 扩展 在 VS Code 中打开扩展市场&#xff08;快捷键 CtrlShiftX 或点击左侧活动栏的扩展图标&#xff09;。搜索“GitHub Copilot Chat”&#xff0c;点击安装。安装完成后需登录 GitHub 账户并授权 Copilot 权限。确保已订阅 GitHub Copilot 服务&am…

bash 脚本比较 100 个程序运行时间,精确到毫秒,脚本

脚本如下&#xff1a; #!/bin/bash# 设置测试次数 NUM_TESTS100 # 设置要测试的程序路径 PROGRAM"./your_program" # 替换为你的程序路径 # 设置程序参数&#xff08;如果没有参数则留空&#xff09; ARGS"" # 例如: "input.txt output.txt"#…

【Linux学习】Linux安装并配置Redis

安装Redis在Linux系统上安装Redis可以通过包管理器或源码编译两种方式进行。以下是两种方法的详细步骤。使用包管理器安装Redis&#xff08;以Ubuntu为例&#xff09;&#xff1a;sudo apt update sudo apt install redis-server通过源码编译安装Redis&#xff1a;wget https:/…

redis每种数据结构对应的底层数据结构原理

Redis 的每种数据结构(String、List、Hash、Set、Sorted Set)在底层都采用了不同的实现方式,根据数据规模和特性动态选择最优的编码(encoding)以节省内存和提高性能。以下是详细原理分析: 1. String(字符串) 底层实现: int:当存储整数值且可用 long 表示时,直接使用…

WPF控件大全:核心属性详解

WPF常用控件及核心属性 以下是WPF开发中最常用的控件及其关键属性&#xff08;按功能分类&#xff09;&#xff1a; 基础布局控件 Grid&#xff08;网格布局&#xff09; RowDefinitions&#xff1a;行定义集合&#xff08;如Height"Auto"&#xff09;ColumnDefinit…

马斯克脑机接口(Neuralink)技术进展,已经实现瘫痪患者通过BCI控制电脑、玩视频游戏、学习编程,未来盲人也能恢复视力了

目录 图片总结文字版总结1. 核心目标与愿景1.1 增强人类能力1.2 解决脑部疾病1.3 理解意识1.4 应对AI风险 2. 技术进展与产品2.1 Telepathy&#xff08;意念操控&#xff09;功能与目标技术细节参与者案例 2.2 Blindsight&#xff08;视觉恢复&#xff09;**功能与目标**技术细…

Vuex身份认证

虽说上一节我们实现了登录功能&#xff0c;但是实际上还是可以通过浏览器的地址来跳过登录访问到后台&#xff0c;这种可有可无的登录功能使得系统没有安全性&#xff0c;而且没有意义 为了让登录这个功能有意义&#xff0c;我们应该&#xff1a; 应当在用户登录成功之后给用户…

springboot中使用线程池

1.什么场景下使用线程池&#xff1f; 在异步的场景下&#xff0c;可以使用线程池 不需要同步等待&#xff0c; 不需要管上一个方法是否执行完毕&#xff0c;你当前的方法就可以立即执行 我们来模拟一下&#xff0c;在一个方法里面执行3个子任务&#xff0c;不需要相互等待 …

Flask+LayUI开发手记(十):构建统一的选项集合服务

作为前端最主要的组件&#xff0c;无论是layui-table表格还是layui-form表单&#xff0c;其中都涉及到选项列的处理。如果是普通编程&#xff0c;一个任务对应一个程序&#xff0c;自然可以就事论事地单对单处理&#xff0c;前后端都配制好选项&#xff0c;手工保证两者的一致性…

redis的数据初始化或增量更新的方法

做系统开发的时候&#xff0c;经常需要切换环境&#xff0c;做一些数据的初始化的工作&#xff0c;而redis的初始化&#xff0c;假如通过命令来执行&#xff0c;又太复杂&#xff0c;因为redis有很多种数据类型&#xff0c;全部通过敲击命令来初始化的话&#xff0c;打的命令实…

【PaddleOCR】OCR表格识别数据集介绍,包含PubTabNet、好未来表格识别、WTW中文场景表格等数据,持续更新中......

&#x1f9d1; 博主简介&#xff1a;曾任某智慧城市类企业算法总监&#xff0c;目前在美国市场的物流公司从事高级算法工程师一职&#xff0c;深耕人工智能领域&#xff0c;精通python数据挖掘、可视化、机器学习等&#xff0c;发表过AI相关的专利并多次在AI类比赛中获奖。CSDN…

sparkjar任务运行

mainclass&#xff1a; test.sparkjar.SparkJarTest

Web攻防-文件下载文件读取文件删除目录遍历路径穿越

知识点&#xff1a; 1、WEB攻防-文件下载&读取&删除-功能点&URL 2、WEB攻防-目录遍历&穿越-功能点&URL 黑盒分析&#xff1a; 1、功能点 文件上传&#xff0c;文件下载&#xff0c;文件删除&#xff0c;文件管理器等地方 2、URL特征 文件名&#xff1a; d…

使用LIMIT + OFFSET 分页时,数据重复的风险

在使用 LIMIT OFFSET 分页时&#xff0c;数据重复的风险不仅与排序字段的唯一性有关&#xff0c;还与数据变动&#xff08;插入、删除、更新&#xff09;密切相关。以下是详细分析&#xff1a; 一、数据变动如何导致分页异常 1. 插入新数据 场景&#xff1a;用户在浏览第 1 页…

Excel 数据透视表不够用时,如何处理来自多个数据源的数据?

当数据透视表感到“吃力”时&#xff0c;我们该怎么办&#xff1a; 数据量巨大&#xff1a;Excel工作表有104万行的限制&#xff0c;当有几十万行数据时&#xff0c;透视表和公式就会变得非常卡顿。数据来源多样&#xff1a;数据分散在多个Excel文件、CSV文件、数据库甚至网页…

cf(1034)Div3(补题A B C D E F)

哈&#xff0c;这个比赛在开了不久之后&#xff0c;不知道为啥卡了差不多20来分钟&#xff0c;后面卡着卡着就想睡觉了。实在是太困了.... 题目意思&#xff1a; Alice做一次操作&#xff0c;删除任意数字a,而Bob做一次操作删除b使得ab对4取余是3。 获胜条件&#xff0c;有人…

浏览器与服务器的交互

浏览器地址栏输入URL&#xff08;网址​​&#xff09; ​​​​(1) 服务器进行URL解析​​&#xff1a;验证URL格式&#xff0c;提取协议、域名等 ​​​​(2) 服务器进行DNS查询​​&#xff1a;将域名转换为IP地址&#xff08;可能涉及缓存或DNS预取&#xff09; ​​​​…

Spring Boot中POST请求参数校验的实战指南

在现代的Web开发中&#xff0c;数据校验是确保应用程序稳定性和安全性的关键环节。Spring Boot提供了强大而灵活的校验机制&#xff0c;能够帮助开发者轻松地对POST请求参数进行校验。本文将详细介绍如何在Spring Boot中实现POST请求参数的校验&#xff0c;并通过具体的代码示例…