力扣第98题 验证二叉搜索树 c++ 与上一篇文章相似

news2024/11/15 11:11:50

题目

98. 验证二叉搜索树

中等

相关标签

树  深度优先搜索  二叉搜索树  二叉树

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

有效 二叉搜索树定义如下:

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

示例 1:

输入:root = [2,1,3]
输出:true

示例 2:

输入:root = [5,1,4,null,null,3,6]
输出:false
解释:根节点的值是 5 ,但是右子节点的值是 4 。

提示:

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

思路和解题方法

第一种方法:

  • 首先定义一个辅助函数 traversal,用于实现中序遍历二叉树,并将节点值存储在一个数组 vec 中。
  • 然后,调用 traversal 函数来遍历二叉树并按顺序存储节点值。
  • 最后,检查数组 vec 是否按照升序排列,若是则为合法的二叉搜索树,返回 true;否则返回 false。

第二种方法:

  • 定义一个全局变量 maxVal,初始值设为 LONG_MIN
  • 使用递归函数 isValidBST 对二叉树进行深度优先搜索。
  • 在搜索过程中,先递归遍历左子树,然后判断当前节点的值是否大于 maxVal,若是则更新 maxVal 为当前节点的值,表示已经访问了当前节点,继续遍历右子树。
  • 若出现当前节点的值小于等于 maxVal,则说明不满足二叉搜索树的要求,返回 false。
  • 如果整个搜索过程中没有出现不满足要求的情况,即二叉树是合法的二叉搜索树,返回 true。

第三种方法:

  • 定义一个指针 pre 用于记录前一个访问的节点。
  • 使用递归函数 isValidBST 对二叉树进行中序遍历。
  • 在中序遍历过程中,先递归遍历左子树,然后判断当前节点的值是否大于前一个节点的值,若不满足这个条件则返回 false。
  • 如果整个中序遍历过程中都满足当前节点值大于前一个节点值的条件,则说明二叉树是合法的二叉搜索树,返回 true。

复杂度

        时间复杂度:

                O(n)

时间复杂度:O(n),其中 nnn 为二叉树的节点个数。二叉树的每个节点最多被访问一次,因此时间复杂度为 O(n)。

        空间复杂度

                O(n)

空间复杂度:O(n),其中 nnn 为二叉树的节点个数。栈最多存储 nnn 个节点,因此需要额外的 O(n)的空间。

c++ 代码一

// 中序遍历二叉树,将节点值按顺序存储在vec数组中
void traversal(TreeNode* root) {
    if(root == NULL) return ;
    // 遍历左子树
    traversal(root->left);
    // 将当前节点值加入数组
    vec.push_back(root->val);
    // 遍历右子树
    traversal(root->right);
}

bool isValidBST(TreeNode* root) {
    // 清空数组
    vec.clear();
    // 进行中序遍历
    traversal(root);
    // 检查数组是否按升序排列
    for(int i = 1; i < vec.size(); i++) {
        // 若出现逆序,则不是二叉搜索树
        if(vec[i] <= vec[i-1]) return false;
    }
    // 数组按升序排列,是二叉搜索树
    return true;
}

c++代码二

long long maxVal = LONG_MIN; // 因为后台测试数据中有int最小值

bool isValidBST(TreeNode* root) {
    // 到达叶节点,返回true,表示是一个合法的搜索二叉树
    if(root == NULL) return true;
    
    // 先递归处理左子树
    bool left = isValidBST(root->left);
    
    // 判断当前节点值是否大于maxVal
    if(maxVal < root->val) 
        maxVal = root->val;
    else 
        return false;
    
    // 再递归处理右子树
    bool right = isValidBST(root->right);
    
    // 左右子树都是合法的搜索二叉树,整棵树才是合法的
    return left && right;
}

c++代码三

TreeNode* pre = NULL; // 用来记录前一个节点

bool isValidBST(TreeNode* root) {
    // 到达叶节点,返回true,表示是一个合法的搜索二叉树
    if(root == NULL) return true;
    
    // 先递归处理左子树
    bool left = isValidBST(root->left);
    
    // 判断当前节点的值是否大于等于前一个节点的值
    if(pre != NULL && pre->val >= root->val)
        return false;
    
    // 更新pre指针为当前节点
    pre = root;
    
    // 再递归处理右子树
    bool right = isValidBST(root->right);
    
    // 左右子树都是合法的搜索二叉树,整棵树才是合法的
    return left && right;
}

c++迭代法

class Solution {
public:
    bool isValidBST(TreeNode* root) {
        stack<TreeNode*> st;  // 创建一个栈用于存储节点
        TreeNode* cur = root;  // 初始化当前节点为根节点
        TreeNode* pre = NULL;  // 初始化前一个节点为NULL,用于记录前一个节点的值
        
        while (cur != NULL || !st.empty()) {
            if (cur != NULL) {
                st.push(cur);  // 将当前节点入栈
                cur = cur->left;  // 继续遍历左子树, 左
            } else {
                cur = st.top();  // 弹出栈顶节点作为当前节点, 中
                st.pop();
                
                // 判断当前节点值是否小于等于前一个节点的值,如果是,则不是有效的二叉搜索树
                if (pre != NULL && cur->val <= pre->val)
                    return false;
                
                pre = cur;  // 更新前一个节点为当前节点
                
                cur = cur->right;  // 继续遍历右子树, 右
            }
        }
        
        return true;  // 遍历完整棵树后,没有发现非法节点值,返回true表示是一个有效的二叉搜索树
    }
};

觉得有用的话可以点点赞,支持一下。

如果愿意的话关注一下。会对你有更多的帮助。

每天都会不定时更新哦  >人<  。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

淘宝天猫商品历史价格API接口

获取淘宝商品历史价格接口的步骤如下&#xff1a; 注册淘宝开放平台&#xff1a;首先在淘宝开放平台上注册一个账号&#xff0c;并进行登录。创建应用&#xff1a;在淘宝开放平台上创建一个应用&#xff0c;并获取该应用的App Key和App Secret&#xff0c;用于后续的接口调用。…

Python“梦寻”淘宝天猫店铺所有数据接口,淘宝店铺所有商品数据API接口,淘宝API接口申请指南(含代码示例)

获取淘宝店铺所有商品数据的接口可以通过淘宝开放平台获取。 具体操作步骤如下&#xff1a; 在淘宝开放平台注册成为开发者&#xff0c;并创建一个应用&#xff0c;获取到所需的 App Key 和 App Secret 等信息。使用获取到的 App Key 和 App Secret&#xff0c;进行签名和认证…

Android组件通信——Intent(二十三)

1. 认识Intent 1.1 知识点 &#xff08;1&#xff09;了解Intent的主要作用&#xff1b; &#xff08;2&#xff09;掌握Activity程序对Intent操作的支持&#xff1b; &#xff08;3&#xff09;可以使用Intent完成Activity程序间的跳转&#xff0c;也可以通过Intent接收返…

QT基础 QChart绘制折线

目录 1.简单折线 2.数学折线 3.可滑动折线 1.简单折线 //![1] //! 折现段坐标QLineSeries *series new QLineSeries(); //![1]//![2] //! 添加点series->append(0, 6);series->append(2, 4);series->append(3, 8);series->append(7, 4);series->append(10, 5)…

YOLOv7改进: CFP:即插即用的多尺度融合模块,EVC助力小目标检测| 顶刊TIP 2023

💡💡💡本文独家改进:即插即用的多尺度融合模块,EVC助力小目标检测 EVC | 亲测在多个数据集实现暴力涨点,强烈推荐,独家首发; 收录: YOLOv7高阶自研专栏介绍:http://t.csdnimg.cn/tYI0c ✨✨✨前沿最新计算机顶会复现 🚀🚀🚀YOLOv7自研创新结合,轻松搞…

MS31804四通道低边驱动器可pin对pin兼容DRV8804

MS31804TE 是一个具有过流保护功能的四通道低边驱动器。MS31804TE 内置钳位二极管&#xff0c;用来钳制由电感负载续流产生的电压。MS31804TE 可以驱动单极步进电机、直流电机、继电器、螺线管或者其它负载。 散热良好的情况下&#xff0c;MS31804TE 可以提供每个通道最高 2A 的…

整理笔记——二极管

一、什么是二极管 二极管是一种由半导体材料制成的一种具有单向导电性能的电子元器件&#xff0c;二极管的核心是PN结。 加在二极管两端的电压和通关的电流被成为&#xff0c;二极管的伏安特性曲线 ​​​ 二极管的正向特性&#xff1a;起初正向电压较小时&#xff0c;正向电流…

[ACTF2020 新生赛]Exec1

拿到题目&#xff0c;不知道是sql注入还是命令执行漏洞 先ping一下主机 有回显&#xff0c;说明是命令执行漏洞 我们尝试去查看目录 127.0.0.1|ls&#xff0c;发现有回显&#xff0c;目录下面有个index.php的文件 我们之间访问index.php 输入127.0.0.1;cat index.php 发现又…

基于Springboot实现汽车租赁平台管理系统项目【项目源码】

基于Springboot实现汽车租赁平台管理系统演示 JAVA简介 Java主要采用CORBA技术和安全模型&#xff0c;可以在互联网应用的数据保护。它还提供了对EJB&#xff08;Enterprise JavaBeans&#xff09;的全面支持&#xff0c;java servlet API&#xff0c;JSP&#xff08;java ser…

选择适合自身业务的HTTP代理有哪些因素决定?

相信对很多爬虫工作者和数据采集的企业来说&#xff0c;如何选购适合自己业务的HTTP代理是一个特别特别困扰的选题&#xff0c;市面上那么多HTTP代理厂商&#xff0c;好像这家有这些缺点&#xff0c;转头又看到另外一家的缺点&#xff0c;要找一家心仪的仿佛大海捞针。今天我们…

Table.Group系列_第4参数为全局的情况下,利用第五参数进行分组汇总

原始数据: 部门与职位存在于同一列中 实现功能: 根据筛选条件,可对部门或职位进行统计汇总第一列列名根据筛选自动变更,显示当前统计的维度 实现方式: 1. 构建筛选器内容 在任意空白单元格内输入需要筛选的内容 2. 插入"组合框"控件,并进行相应设置 从开发工具…

flex布局在多层嵌套时,内层设置了justify-content: space-between;不生效问题

内层的地址和时间这一行&#xff0c;设置了justify-content: space-between;但并不生效&#xff0c;原因是要在上一层.center 设置 flex:1;&#xff08; 重点&#xff09; 经常忘记&#xff0c;特在此记录一下&#xff0c;以下是代码 <view class"index-card" c…

深度学习基础知识 BatchNorm、LayerNorm、GroupNorm的用法解析

深度学习基础知识 BatchNorm、LayerNorm、GroupNorm的用法解析 1、BatchNorm2、LayerNorm3、GroupNorm用法&#xff1a; BatchNorm、LayerNorm 和 GroupNorm 都是深度学习中常用的归一化方式。 它们通过将输入归一化到均值为 0 和方差为 1 的分布中&#xff0c;来防止梯度消失和…

学生用的台灯哪种比较好?分享专家推荐的学生台灯

对于学生来说&#xff0c;台灯是必不可少的一盏学习照明灯具&#xff0c;它能提供室内不足的光线、亮度&#xff0c;基本每个学生在宿舍、家里都会备着一台。不过台灯也并不是随便挑选一台使用就可以的&#xff0c;很多学生就是因为使用了一些价格低廉、质量安全没有保障的台灯…

【大数据】hadoop安装部署(学习笔记)

一、集群组成概述 Hadoop集群包括两个集群&#xff1a;HDFS集群、YARN集群 两个集群逻辑上分离、通常物理上在一起 两个集群都是标准的主从架构集群 HDFS集群&#xff08;分布式存储&#xff09;&#xff1a; 主角色&#xff1a;NameNode从角色&#xff1a;DataNode主角色…

代码随想录算法训练营第五十八天 |583. 两个字符串的删除操作、72. 编辑距离、编辑距离总结篇

一、583. 两个字符串的删除操作 题目链接/文章讲解/视频讲解&#xff1a;代码随想录 思考&#xff1a; 1.确定dp数组&#xff08;dp table&#xff09;以及下标的含义 dp[i][j]&#xff1a;以i-1为结尾的字符串word1&#xff0c;和以j-1位结尾的字符串word2&#xff0c;想要达…

postman 密码rsa加密登录-2加密密码

上一篇讲了获取公钥&#xff0c;将环境准备好之后&#xff0c;在登录接口的Pre-request Scrip 里&#xff0c;使用公钥进行加密后在正常登录。本文采用的方案是使用第三方模块forge.js来实现加密。 1、环境准备好&#xff0c;系统git 和node都OK。下载forge.js git clone htt…

Java-Atomic原子操作类详解及源码分析,Java原子操作类进阶,LongAdder源码分析

文章目录 一、Java原子操作类概述1、什么是原子操作类2、为什么要用原子操作类3、CAS入门 二、基本类型原子类1、概述2、代码实例 三、数组类型原子类1、概述2、代码实例 四、引用类型原子类1、概述2、AtomicReference3、ABA问题与AtomicStampedReference4、一次性修改&#xf…

SpringBoot (1)

目录 1 入门案例 1.1 环境准备 1.2 编写pom.xml 1.3 编写入口程序 1.4 编写接口 1.5 编写配置 1.6 快速部署 1.6.1 打jar包 1.6.2 部署 1.7 访问接口 2 全注解开发 2.1 常用注解 2.2 属性绑定注解 2.2.1 注册组件 2.2.2 ConfigurationProperties(prefix"te…

AlphaPose Pytorch 代码详解(一):predict

前言 代码地址&#xff1a;AlphaPose-Pytorch版 本文以图像 1.jpg&#xff08;854x480&#xff09;为例对整个预测过程的各个细节进行解读并记录 python demo.py --indir examples/demo --outdir examples/res --save_img1. YOLO 1.1 图像预处理 cv2读取BGR图像 img [480,…