当前位置: 首页 > news >正文

《程序员面试金典(第6版)》面试题 04.05. 合法二叉搜索树

题目描述

实现一个函数,检查一棵二叉树是否为二叉搜索树。

示例 1:
输入:

    2/ \1   3

输出: true

示例 2:
输入:

    5/ \1   4/ \3   6

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

解题思路与代码

使用额外数据结构 + 中序遍历

这应该是最简单,并且最容易理解的一种做法了。
由二叉搜索树的性质可知,二叉搜索树的左边节点小于中间节点,中间节点小于右边节点。由这一特性我们可以得知,二叉搜索树的中序遍历是一个有序的数组。

我们只需要检查这个数组是否有序,就能判断出这个二叉树是否是二叉搜索树。

具体实现请看代码:

/*** Definition for a binary tree node.* struct TreeNode {*     int val;*     TreeNode *left;*     TreeNode *right;*     TreeNode(int x) : val(x), left(NULL), right(NULL) {}* };*/
class Solution {
public:bool isValidBST(TreeNode* root) {vector<int> result;inorderTraversal(root,result);int size = result.size();for(int i = 1; i < size; ++i )if(result[i-1] >= result[i]) return false;return true;}void inorderTraversal(TreeNode* root,vector<int>& vec){if(root == nullptr) return ;inorderTraversal(root->left,vec);vec.push_back(root->val);inorderTraversal(root->right,vec);}
};

在这里插入图片描述

复杂度分析

时间复杂度:O(n),n是这个二叉树的节点数目。我们要遍历这个二叉树的每一个节点。
空间复杂度:O(n),n是这个二叉树的节点数目,我们要将n个元素压入vector中。此外,函数的递归调用会使用一定的系统栈空间,但是由于递归深度不会超过二叉树的高度,所以可以认为空间复杂度也是 O(n) 级别的。

中序遍历不使用额外数据结构

这种做法,就是稍稍的将我刚刚的那种做法升级了一下,我们使用一个前驱指针,去记录中序遍历的前一个节点。那么中序遍历是先遍历左子树然后遍历中间节点,再去遍历右子树的。所以我们只需要一直去判断,这个前驱指针的值是否一种小于中间节点的值就行了。

具体实现请看代码:

class Solution {
public:TreeNode* pre = nullptr;bool isValidBST(TreeNode* root) {if(root == nullptr) return true;if(!isValidBST(root->left)) return false;if(pre != nullptr && pre->val >= root->val) return false;pre = root;if(!isValidBST(root->right)) return false;return true;}
};

注意:这个代码中不太容易想出来的代码在于这一行:if(!isValidBST(root->left)) return false; 我们的递归逻辑是在这个if判断里面的。这个递归就会把我们一直带到左叶子节点。然后再逐层的返回判断。
在这里插入图片描述

复杂度分析:

时间复杂度:O(n),n为节点的个数。不管怎样我们都要遍历这个树中的所有节点。
空间复杂度:O(h),其中 h 是二叉树的高度。最坏情况下(即二叉树退化为链表时),递归调用深度达到 n,此时栈的空间复杂度为 O(n);平均情况下树的高度是 log n,空间复杂度是 O(log n)。

这个代码优化了空间复杂度,因为没有用到额外的数据结构。但是执行代码所需要的时间缺增加了。这是因为我们每次递归都要做许多的判断。而上一次的代码只需要在for循环里做少量判断就行了。

总结

这道题不算太难。只要能及时的想起二叉搜索树的性质,就能轻松解题。

http://www.lryc.cn/news/39408.html

相关文章:

  • Nginx 反向代理技术梳理
  • 华为OD机试 - 整数编码(Java) | 机试题+算法思路+考点+代码解析 【2023】
  • 蓝桥杯冲击01 - 质数篇
  • 【WEB前端进阶之路】 HTML 全路线学习知识点梳理(下)
  • MySQL索引分类
  • 会声会影2023最新版图文安装详细教程
  • Java中的反射
  • STM32入门笔记(03):STM32F103C8T6定时器的输入捕获模式和编码器模式(SPL库函数版)
  • 《网络安全》零基础教程-适合小白科普
  • 微信小程序语言与web开发语言的区别
  • 【2022-09-14】米哈游秋招笔试三道编程题
  • 云监控能力介绍
  • HTML 文档类型
  • 【UML】软件设计说明书 (完结)
  • MATLAB——FFT(快速傅里叶变换)
  • 力扣-进店却未进行过交易的顾客
  • 一文解决vscode中借助CMake配置使用Opencv过程中的所有问题
  • Golang每日一练(leetDay0004)
  • 手机忘记密码解锁的 6 大软件方法
  • MySQL数据库的基础语法总结(1)
  • Linux之进程创建
  • DCL 管理用户与权限控制
  • 如何使用 Python 检测和识别车牌(附 Python 代码)
  • [Python题解] CodeForces 1804 D. Accommodation
  • 【设计模式】访问者模式
  • 蓝桥杯刷题冲刺 | 倒计时27天
  • RV1126_python人脸识别Retinaface+MobilefaceNet
  • HBase---HBase基础语法
  • 2023年,PMP有多少含金量呢?
  • vue动态路由