算法(TS):验证二叉搜索树

给你一个二叉树的根节点 root ,判断其是否是一个有效的二叉搜索树。有效 二叉搜索树定义如下:

  • 节点的左子树只包含小于当前节点的数。
  • 节点的右子树只包含大于当前节点的数。
  • 所有左子树和右子树自身必须也是二叉搜索树。

示例

示例一

示例一是有效的二叉搜索树

示例二

示例二不是有效的二叉搜索树,因为根节点的值是 5 ,但是右子节点的值是 4 。

提示:

  • 树中节点数目范围在[1, 104] 内
  • -231 <= Node.val <= 231 - 1****

解答

解答一

有效的二叉搜索树中每一个节点都有一个最大取值和最小取值,以示例二来说,假设我们要将它变成有效二叉搜索树,那么,顶层节点【5】的取值范围是(-Infinity,Infinity),第一层的左节点【1】的取值范围是(-Infinity,5),第一层的右节点【4】的取值范围是(5,Infinity);接下来分享第三层,第三层【3】的取值范围是(【4】的最小取值,【4】的值),第三层【6】的取值范围是(【4】的值,【4】的最大取值,即Infinity)。

使用递归解法如下:

kotlin 复制代码
/**
 * Definition for a binary tree node.
 * class TreeNode {
 *     val: number
 *     left: TreeNode | null
 *     right: TreeNode | null
 *     constructor(val?: number, left?: TreeNode | null, right?: TreeNode | null) {
 *         this.val = (val===undefined ? 0 : val)
 *         this.left = (left===undefined ? null : left)
 *         this.right = (right===undefined ? null : right)
 *     }
 * }
 */

function isValidBST(root: TreeNode | null): boolean {
    const isValid = (root:TreeNode | null,minVal: number,maxVal: number) => {
        if (!root) return true
        if (root.val <= minVal || root.val >= maxVal) {
            return false
        }
        return isValid(root.left,minVal,root.val) && isValid(root.right,root.val,maxVal)
    }
    return isValid(root,-Infinity,Infinity)
};

时间复杂度O(n),空间复杂度O(n)

解答二

进行中序优先遍历,如果是有效的二叉搜索树,那么上一个节点的值一定小于当前节点的值,否则不是有效的二叉搜索树。

ini 复制代码
/**
 * Definition for a binary tree node.
 * class TreeNode {
 *     val: number
 *     left: TreeNode | null
 *     right: TreeNode | null
 *     constructor(val?: number, left?: TreeNode | null, right?: TreeNode | null) {
 *         this.val = (val===undefined ? 0 : val)
 *         this.left = (left===undefined ? null : left)
 *         this.right = (right===undefined ? null : right)
 *     }
 * }
 */

function isValidBST(root: TreeNode | null): boolean {
    let prevVal = -Infinity
    const stack: TreeNode[] = []
    while(root || stack.length) {
        while(root) {
            stack.push(root)
            root = root.left
        }

        root = stack.pop()
        if (root.val <= prevVal) {
            return false
        }

        prevVal = root.val
        root = root.right
    }

    return true
};

时间复杂度O(n),空间复杂度O(n)

相关推荐
excel5 分钟前
ES6 中函数的双重调用方式:fn() 与 fn\...``
前端
可乐爱宅着17 分钟前
全栈框架next.js入手指南
前端·next.js
你的人类朋友2 小时前
什么是API签名?
前端·后端·安全
会豪4 小时前
Electron-Vite (一)快速构建桌面应用
前端
中微子4 小时前
React 执行阶段与渲染机制详解(基于 React 18+ 官方文档)
前端
唐某人丶4 小时前
教你如何用 JS 实现 Agent 系统(2)—— 开发 ReAct 版本的“深度搜索”
前端·人工智能·aigc
中微子4 小时前
深入剖析 useState产生的 setState的完整执行流程
前端
遂心_4 小时前
JavaScript 函数参数传递机制:一道经典面试题解析
前端·javascript
小徐_23334 小时前
uni-app vue3 也能使用 Echarts?Wot Starter 是这样做的!
前端·uni-app·echarts
RoyLin4 小时前
TypeScript设计模式:适配器模式
前端·后端·node.js