经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 程序设计 » C++ » 查看文章
C++实现LeetCode(98.验证二叉搜索树)
来源:jb51  时间:2021/7/19 19:54:08  对本文有异议

[LeetCode] 98. Validate Binary Search Tree 验证二叉搜索树

Given a binary tree, determine if it is a valid binary search tree (BST).

Assume a BST is defined as follows:

  • The left subtree of a node contains only nodes with keys less than the node's key.
  • The right subtree of a node contains only nodes with keys greater than the node's key.
  • Both the left and right subtrees must also be binary search trees.

Example 1:

Input:
2
/ \
1   3
Output: true

Example 2:

    5
/ \
1   4
/ \
3   6
Output: false
Explanation: The input is: [5,1,4,null,null,3,6]. The root node's value
is 5 but its right child's value is 4.

这道验证二叉搜索树有很多种解法,可以利用它本身的性质来做,即左<根<右,也可以通过利用中序遍历结果为有序数列来做,下面我们先来看最简单的一种,就是利用其本身性质来做,初始化时带入系统最大值和最小值,在递归过程中换成它们自己的节点值,用long代替int就是为了包括int的边界条件,代码如下:

C++ 解法一:

  1. // Recursion without inorder traversal
  2. class Solution {
  3. public:
  4. bool isValidBST(TreeNode* root) {
  5. return isValidBST(root, LONG_MIN, LONG_MAX);
  6. }
  7. bool isValidBST(TreeNode* root, long mn, long mx) {
  8. if (!root) return true;
  9. if (root->val <= mn || root->val >= mx) return false;
  10. return isValidBST(root->left, mn, root->val) && isValidBST(root->right, root->val, mx);
  11. }
  12. };

Java 解法一:

  1. public class Solution {
  2. public boolean isValidBST(TreeNode root) {
  3. if (root == null) return true;
  4. return valid(root, Long.MIN_VALUE, Long.MAX_VALUE);
  5. }
  6. public boolean valid(TreeNode root, long low, long high) {
  7. if (root == null) return true;
  8. if (root.val <= low || root.val >= high) return false;
  9. return valid(root.left, low, root.val) && valid(root.right, root.val, high);
  10. }
  11. }

这题实际上简化了难度,因为有的时候题目中的二叉搜索树会定义为左<=根<右,而这道题设定为一般情况左<根<右,那么就可以用中序遍历来做。因为如果不去掉左=根这个条件的话,那么下边两个数用中序遍历无法区分:

   20       20
/           \
20           20

它们的中序遍历结果都一样,但是左边的是 BST,右边的不是 BST。去掉等号的条件则相当于去掉了这种限制条件。下面来看使用中序遍历来做,这种方法思路很直接,通过中序遍历将所有的节点值存到一个数组里,然后再来判断这个数组是不是有序的,代码如下:

C++ 解法二:

  1. // Recursion
  2. class Solution {
  3. public:
  4. bool isValidBST(TreeNode* root) {
  5. if (!root) return true;
  6. vector<int> vals;
  7. inorder(root, vals);
  8. for (int i = 0; i < vals.size() - 1; ++i) {
  9. if (vals[i] >= vals[i + 1]) return false;
  10. }
  11. return true;
  12. }
  13. void inorder(TreeNode* root, vector<int>& vals) {
  14. if (!root) return;
  15. inorder(root->left, vals);
  16. vals.push_back(root->val);
  17. inorder(root->right, vals);
  18. }
  19. };

Java 解法二:

  1. public class Solution {
  2. public boolean isValidBST(TreeNode root) {
  3. List<Integer> list = new ArrayList<Integer>();
  4. inorder(root, list);
  5. for (int i = 0; i < list.size() - 1; ++i) {
  6. if (list.get(i) >= list.get(i + 1)) return false;
  7. }
  8. return true;
  9. }
  10. public void inorder(TreeNode node, List<Integer> list) {
  11. if (node == null) return;
  12. inorder(node.left, list);
  13. list.add(node.val);
  14. inorder(node.right, list);
  15. }
  16. }

下面这种解法跟上面那个很类似,都是用递归的中序遍历,但不同之处是不将遍历结果存入一个数组遍历完成再比较,而是每当遍历到一个新节点时和其上一个节点比较,如果不大于上一个节点那么则返回 false,全部遍历完成后返回 true。代码如下:

C++ 解法三:

  1. class Solution {
  2. public:
  3. bool isValidBST(TreeNode* root) {
  4. TreeNode *pre = NULL;
  5. return inorder(root, pre);
  6. }
  7. bool inorder(TreeNode* node, TreeNode*& pre) {
  8. if (!node) return true;
  9. bool res = inorder(node->left, pre);
  10. if (!res) return false;
  11. if (pre) {
  12. if (node->val <= pre->val) return false;
  13. }
  14. pre = node;
  15. return inorder(node->right, pre);
  16. }
  17. };

当然这道题也可以用非递归来做,需要用到栈,因为中序遍历可以非递归来实现,所以只要在其上面稍加改动便可,代码如下:

C++ 解法四:

  1. class Solution {
  2. public:
  3. bool isValidBST(TreeNode* root) {
  4. stack<TreeNode*> s;
  5. TreeNode *p = root, *pre = NULL;
  6. while (p || !s.empty()) {
  7. while (p) {
  8. s.push(p);
  9. p = p->left;
  10. }
  11. p = s.top(); s.pop();
  12. if (pre && p->val <= pre->val) return false;
  13. pre = p;
  14. p = p->right;
  15. }
  16. return true;
  17. }
  18. };

Java 解法四:

  1. public class Solution {
  2. public boolean isValidBST(TreeNode root) {
  3. Stack<TreeNode> s = new Stack<TreeNode>();
  4. TreeNode p = root, pre = null;
  5. while (p != null || !s.empty()) {
  6. while (p != null) {
  7. s.push(p);
  8. p = p.left;
  9. }
  10. p = s.pop();
  11. if (pre != null && p.val <= pre.val) return false;
  12. pre = p;
  13. p = p.right;
  14. }
  15. return true;
  16. }
  17. }

最后还有一种方法,由于中序遍历还有非递归且无栈的实现方法,称之为 Morris 遍历,可以参考博主之前的博客 Binary Tree Inorder Traversal,这种实现方法虽然写起来比递归版本要复杂的多,但是好处在于是 O(1) 空间复杂度,参见代码如下:

C++ 解法五:

  1. class Solution {
  2. public:
  3. bool isValidBST(TreeNode *root) {
  4. if (!root) return true;
  5. TreeNode *cur = root, *pre, *parent = NULL;
  6. bool res = true;
  7. while (cur) {
  8. if (!cur->left) {
  9. if (parent && parent->val >= cur->val) res = false;
  10. parent = cur;
  11. cur = cur->right;
  12. } else {
  13. pre = cur->left;
  14. while (pre->right && pre->right != cur) pre = pre->right;
  15. if (!pre->right) {
  16. pre->right = cur;
  17. cur = cur->left;
  18. } else {
  19. pre->right = NULL;
  20. if (parent->val >= cur->val) res = false;
  21. parent = cur;
  22. cur = cur->right;
  23. }
  24. }
  25. }
  26. return res;
  27. }
  28. };

到此这篇关于C++实现LeetCode(98.验证二叉搜索树)的文章就介绍到这了,更多相关C++实现验证二叉搜索树内容请搜索w3xue以前的文章或继续浏览下面的相关文章希望大家以后多多支持w3xue!

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号