牛客题解 | 判断是不是二叉搜索树

news/2025/2/25 19:15:06/文章来源:https://www.cnblogs.com/wc529065/p/18737033

题目

题目链接

题目主要信息:
  • 判断给定的一棵二叉树是否是二叉搜索树
  • 二叉搜索树每个左子树元素小于根节点,每个右子树元素大于根节点,中序遍历为递增序
举一反三:

学习完本题的思路你可以解决如下题目:

BM30. 二叉搜索树与双向链表

BM37. 二叉搜索树的最近公共祖先

方法一:递归(推荐使用)

知识点1:二叉树递归

递归是一个过程或函数在其定义或说明中有直接或间接调用自身的一种方法,它通常把一个大型复杂的问题层层转化为一个与原问题相似的规模较小的问题来求解。因此递归过程,最重要的就是查看能不能讲原本的问题分解为更小的子问题,这是使用递归的关键。

而二叉树的递归,则是将某个节点的左子树、右子树看成一颗完整的树,那么对于子树的访问或者操作就是对于原树的访问或者操作的子问题,因此可以自我调用函数不断进入子树。

知识点2:二叉搜索树

二叉搜索树是一种特殊的二叉树,它的每个节点值大于它的左子节点,且大于全部左子树的节点值,小于它右子节点,且小于全部右子树的节点值。因此二叉搜索树一定程度上算是一种排序结构。

思路:

二叉搜索树的特性就是中序遍历是递增序。既然是判断是否是二叉搜索树,那我们可以使用中序递归遍历。只要之前的节点是二叉树搜索树,那么如果当前的节点小于上一个节点值那么就可以向下判断。**只不过在过程中我们要求反退出*。比如一个链表1->2->3->4,只要for循环遍历如果中间有不是递增的直接返回false即可。

if(root.val < pre)return false;

具体做法:

  • step 1:首先递归到最左,初始化maxLeft与pre。
  • step 2:然后往后遍历整棵树,依次连接pre与当前节点,并更新pre。
  • step 3:左子树如果不是二叉搜索树返回false。
  • step 4:判断当前节点是不是小于前置节点,更新前置节点。
  • step 5:最后由右子树的后面节点决定。

Java实现代码:

import java.util.*;
public class Solution {int pre = Integer.MIN_VALUE;//中序遍历public boolean isValidBST (TreeNode root) { if (root == null)return true;//先进入左子树if(!isValidBST(root.left)) return false;if(root.val < pre)return false;//更新最值pre = root.val;//再进入右子树return isValidBST(root.right); }
}

C++实现代码:

class Solution {
public:long pre = INT_MIN;//中序遍历bool isValidBST(TreeNode* root) {if(root == NULL)return true;//先进入左子树if(!isValidBST(root->left)) return false;if(root->val <= pre)return false;//更新最值pre = root->val;  //再进入右子树if(!isValidBST(root->right))  return false;return true;}
};

Python实现代码

import sys
class Solution:pre = -sys.maxsize - 1def isValidBST(self , root: TreeNode) -> bool:if not root:return True# 先进入左子树if not self.isValidBST(root.left): return Falseif(root.val <= self.pre):return False# 更新最值self.pre = root.val  # 再进入右子树if not self.isValidBST(root.right):  return Falsereturn True

复杂度分析:

  • 时间复杂度:\(O(n)\),其中\(n\)为二叉树的节点数,最坏遍历整个二叉树所有节点
  • 空间复杂度:\(O(n)\),最坏情况二叉树退化为链表,递归栈深度为\(n\)
方法二:非递归(扩展思路)

知识点:栈

栈是一种仅支持在表尾进行插入和删除操作的线性表,这一端被称为栈顶,另一端被称为栈底。元素入栈指的是把新元素放到栈顶元素的上面,使之成为新的栈顶元素;元素出栈指的是从一个栈删除元素又称作出栈或退栈,它是把栈顶元素删除掉,使其相邻的元素成为新的栈顶元素。

思路:

我们也可以利用栈来代替递归。如果一棵二叉树,对于每个根节点都优先访问左子树,那结果是什么?从根节点开始不断往左,第一个被访问的肯定是最左边的节点,
然后访问该节点的右子树,最后向上回到父问题。因为每次访问最左的元素不止对一整棵二叉树成立,而是对所有子问题都成立,因此循环的时候自然最开始都是遍历到最左:

//直到没有左节点
while(head != null){   s.push(head);head = head.left;
}

然后访问,然后再进入右子树,我们可以用栈来实现回归父问题。

具体做法:

  • step 1:优先判断树是否为空,空树不遍历。
  • step 2:准备一个数组记录中序遍历的结果。
  • step 3:准备辅助栈,当二叉树节点为空了且栈中没有节点了,我们就停止访问。
  • step 4:从根节点开始,每次优先进入每棵的子树的最左边一个节点,我们将其不断加入栈中,用来保存父问题。
  • step 5:到达最左后,可以开始访问,如果它还有右节点,则将右边也加入栈中,之后右子树的访问也是优先到最左。
  • step 6:遍历数组,依次比较相邻两个元素是否为递增序。

图示:

alt

Java实现代码:

import java.util.*;
public class Solution {public boolean isValidBST(TreeNode root){//设置栈用于遍历Stack<TreeNode> s = new Stack<TreeNode>(); TreeNode head = root;//记录中序遍历的数组ArrayList<Integer> sort = new ArrayList<Integer>(); while(head != null || !s.isEmpty()){//直到没有左节点while(head != null){   s.push(head);head = head.left;}head = s.pop();//访问节点sort.add(head.val); //进入右边head = head.right; }//遍历中序结果for(int i = 1; i < sort.size(); i++){ //一旦有降序,则不是搜索树if(sort.get(i - 1) > sort.get(i)) return false;}return true;}
}

C++实现代码:

class Solution {
public:bool isValidBST(TreeNode* root) {//设置栈用于遍历stack<TreeNode*> s; TreeNode* head = root;vector<int> sort; //记录中序遍历的数组while(head != NULL || !s.empty()){//直到没有左节点while(head != NULL){   s.push(head);head = head->left;}head = s.top();s.pop();//访问节点sort.push_back(head->val);head = head->right;}//遍历中序结果for(int i = 1; i < sort.size(); i++){ //一旦有降序,则不是搜索树if(sort[i - 1] > sort[i]) return false;}return true;}
};

Python实现代码

class Solution:def isValidBST(self , root: TreeNode) -> bool:# 设置栈用于遍历 和 记录中序遍历的数组s ,sort = [], [] head = rootwhile head or s:# 直到没有左节点while head:   s.append(head)head = head.lefthead = s[-1]s.pop()# 访问节点sort.append(head.val) head = head.right# 遍历中序结果for i in range(1, len(sort)):  # 一旦有降序,则不是搜索树if sort[i - 1] > sort[i]: return Falsereturn True

复杂度分析:

  • 时间复杂度:\(O(n)\),其中\(n\)为二叉树的节点数,遍历整个二叉树后又遍历数组
  • 空间复杂度:\(O(n)\),辅助栈及辅助数组的空间最坏为\(O(n)\)

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

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

相关文章

牛客题解 | 判断是不是完全二叉树

牛客题库题解题目 题目链接 题目主要信息:判断给定二叉树是否为完全二叉树 首先我们需要知道什么是完全二叉树:叶子节点只能出现在最下层和次下层,且最下层的叶子节点集中在树的左部。 需要注意的是,满二叉树肯定是完全二叉树,而完全二叉树不一定是满二叉树。举一反三: 学…

linux下安装 elasticsearch

一、基础环境 操作系统环境:Red Hat Enterprise Linux Server release 6.4 (Santiago) ES版本:elasticsearch-7.8.0-linux-x86_64.tar.gz Jdk:Java(TM) SE Runtime Environment (build 1.8.0_144-b01) 二、安装 1、上传安装包到/opt目录下 2、解压cd /opt # tar -zxvf elas…

mysql 啥样的索引能提高查询性能呢?

前言 在前面几章中,我们知道了页里面是如何存储的,页又是如何编排的。 这样我们知道了,如何定位到页,如何定位到行了,这些对我们索引的了解非常有帮助的。 知道这些后,那么我们如何利用索引查询呢? 也就是说我们如何利用这种数据结构呢? 是不是全部的查询都能通过索引去…

06 常用损失函数介绍

在前文中我们使用的损失函数都是均方误差(MSE,Mean Squared Error),本篇介绍一些其他的损失函数形式,以及他们的不同用途。 1. 回归任务常用损失函数 1.1 均方误差(MSE, Mean Squared Error) 均方误差(MSE)是回归任务中最常用的损失函数之一,用于衡量模型预测值与真实…

啦啦啦啦啦啦啦啦啦

啦啦啦 啦啦啦啦啦啦啦啦啦 ABC221G 神秘题,将坐标轴转 \(45\),然后 bitset 优化背包,记录路径把刚刚被更新的找出来,然后 _Find_next,每个点只会记一次。 AGC050a 神秘题,想到 \(\log\),然后发现一下位置 \(x\) 走十次能到的区间是 \([1024x,1024x+1023]\),区间长度够…

破解 vLLM + DeepSeek 规模化部署的“不可能三角”

通过 FC GPU 预留实例的闲置计费功能,企业用户能在充分利用 vLLM 的强大功能的同时找到成本、性能、稳定的最佳平衡点,并保持开发和运维的高效性。无论是将 FC vLLM 函数直接对外提供服务,还是深度集成到现有系统中,或是通过 CAP 还是魔搭来简化部署,都能找到满足您业务需…

条形码编码规则全解析:从黑白条纹到数字世界的转换密码

条形码的编码规则是将字符(数字、字母等)转换为特定黑白条纹或矩阵结构的标准化方法,核心目的是让机器能够快速、准确地识别和解析信息。以下是常见条形码编码规则的简介: 一维条形码编码规则 1. ​基本原理通过不同宽度的黑白条纹​(或空格)组合表示字符。 每个字符对应…

[汽车电子/车联网] CANoe

概述:CANoe CAN 全家桶区别: CANoe vs CanalyzerCANoe和CANalyzer使用方法类似(简直可以说 相同)。 都可用于simulation,区别在于CANalyzer只能模拟单个Node,而CANoe可以同时模拟多个Node。如果入门学习了CANoe,就不用入门学习CANalyzer了。安装指南 安装 CANoeDemo on …

璞华易研PLM荣登软服之家多项榜单,PLM+AI为流程行业提供产品创新引擎

近日,国内知名软件与服务评测平台软服之家发布了多个PLM(Product Lifecycle Management,产品全生命周期管理)榜单,帮助用户了解PLM领域表现卓越的软件产品和服务。在软服之家的多项榜单中,璞华易研PLM凭借其自主研发能力与行业深耕优势,在流程行业、电子信息、装备制造等…

ELK 原理介绍及实践详解

介绍了ELK(Elasticsearch, Logstash, Kibana)在大规模日志管理中的重要性,阐述了ELK解决日志分析的挑战,如日志收集、传输、存储和分析。文章详细讲解了ELK的组成部分,包括Filebeat的工作原理、Logstash的输入、过滤和输出阶段,以及Kibana的分析和可视化功能。此外,还提…

工信部人才交流中心PostgreSQL认证考试 - 聊一下更多精彩

在数字经济高速发展的今天,数据库作为信息基础设施的核心组件,其技术自主性与人才储备已成为国家战略竞争力的关键。工业和信息化部人才交流中心(以下简称“工信人才”)推出PostgreSQL认证考试,不仅是对技术发展趋势的精准响应,更是推动信创产业生态建设、填补数据库人才缺…

单链表与单循环链表的C语言实现

单链表与单循环链表的C语言实现 目录单链表与单循环链表的C语言实现单链表的增删查改单循环链表的增删查改 单链表的增删查改 /*单链表*/#include<stdio.h> #include<stdlib.h>typedef struct Node{int data;struct Node* next; }Node;Node* initList(){Node* node…