文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

C++实现验证二叉搜索树代码

2023-06-20 17:08

关注

本篇内容主要讲解“C++实现验证二叉搜索树代码”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“C++实现验证二叉搜索树代码”吧!

验证二叉搜索树

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

Assume a BST is defined as follows:

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++ 解法一:

// Recursion without inorder traversalclass Solution {public:    bool isValidBST(TreeNode* root) {        return isValidBST(root, LONG_MIN, LONG_MAX);    }    bool isValidBST(TreeNode* root, long mn, long mx) {        if (!root) return true;        if (root->val <= mn || root->val >= mx) return false;        return isValidBST(root->left, mn, root->val) && isValidBST(root->right, root->val, mx);    }};

Java 解法一:

public class Solution {    public boolean isValidBST(TreeNode root) {        if (root == null) return true;        return valid(root, Long.MIN_VALUE, Long.MAX_VALUE);    }    public boolean valid(TreeNode root, long low, long high) {        if (root == null) return true;        if (root.val <= low || root.val >= high) return false;        return valid(root.left, low, root.val) && valid(root.right, root.val, high);    }}

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

   20       20
/           \
20           20

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

C++ 解法二:

// Recursionclass Solution {public:    bool isValidBST(TreeNode* root) {        if (!root) return true;        vector<int> vals;        inorder(root, vals);        for (int i = 0; i < vals.size() - 1; ++i) {            if (vals[i] >= vals[i + 1]) return false;        }        return true;    }    void inorder(TreeNode* root, vector<int>& vals) {        if (!root) return;        inorder(root->left, vals);        vals.push_back(root->val);        inorder(root->right, vals);    }};

Java 解法二:

public class Solution {    public boolean isValidBST(TreeNode root) {        List<Integer> list = new ArrayList<Integer>();        inorder(root, list);        for (int i = 0; i < list.size() - 1; ++i) {            if (list.get(i) >= list.get(i + 1)) return false;        }        return true;    }    public void inorder(TreeNode node, List<Integer> list) {        if (node == null) return;        inorder(node.left, list);        list.add(node.val);        inorder(node.right, list);    }}

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

C++ 解法三:

class Solution {public:    bool isValidBST(TreeNode* root) {        TreeNode *pre = NULL;        return inorder(root, pre);    }    bool inorder(TreeNode* node, TreeNode*& pre) {        if (!node) return true;        bool res = inorder(node->left, pre);        if (!res) return false;        if (pre) {            if (node->val <= pre->val) return false;        }        pre = node;        return inorder(node->right, pre);    }};

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

C++ 解法四:

class Solution {public:    bool isValidBST(TreeNode* root) {        stack<TreeNode*> s;        TreeNode *p = root, *pre = NULL;        while (p || !s.empty()) {            while (p) {                s.push(p);                p = p->left;            }            p = s.top(); s.pop();            if (pre && p->val <= pre->val) return false;            pre = p;            p = p->right;        }        return true;    }};

Java 解法四:

public class Solution {    public boolean isValidBST(TreeNode root) {        Stack<TreeNode> s = new Stack<TreeNode>();        TreeNode p = root, pre = null;        while (p != null || !s.empty()) {            while (p != null) {                s.push(p);                p = p.left;            }            p = s.pop();            if (pre != null && p.val <= pre.val) return false;            pre = p;            p = p.right;        }        return true;    }}

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

C++ 解法五:

class Solution {public:    bool isValidBST(TreeNode *root) {        if (!root) return true;        TreeNode *cur = root, *pre, *parent = NULL;        bool res = true;        while (cur) {            if (!cur->left) {                if (parent && parent->val >= cur->val) res = false;                parent = cur;                cur = cur->right;            } else {                pre = cur->left;                while (pre->right && pre->right != cur) pre = pre->right;                if (!pre->right) {                    pre->right = cur;                    cur = cur->left;                } else {                    pre->right = NULL;                    if (parent->val >= cur->val) res = false;                    parent = cur;                    cur = cur->right;                }            }        }        return res;    }};

到此,相信大家对“C++实现验证二叉搜索树代码”有了更深的了解,不妨来实际操作一番吧!这里是编程网网站,更多相关内容可以进入相关频道进行查询,关注我们,继续学习!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯