审题
给定一个二叉树,判断其是否是一个有效的二叉搜索树。
假设一个二叉搜索树具有如下特征:
节点的左子树只包含小于当前节点的数。
节点的右子树只包含大于当前节点的数。
所有左子树和右子树自身必须也是二叉搜索树。
示例 1:
输入:
2
/ \
1 3
输出: true
示例 2:
输入:
5
/ \
1 4
/ \
3 6
输出: false
解释: 输入为: [5,1,4,null,null,3,6]。
根节点的值为 5 ,但是其右子节点值为 4 。
看到这道题,没想法我只能说,不知道要怎么深度遍历,因为之前树这里的知识就掌握的不是很好,因此这次,我们直接看题解,看了之后,两点收获吧:
- 在方案一中,我们利用二叉搜索树的性质,左子树的所有值都小于根节点,对于每次递归更新最小值或者最大值,随着每次递归都是一次二叉搜索树条件的确认,最后生成的二叉搜索树自然就是真正的二叉搜索树。
- 在方案二中,我们运用了二叉搜索树的一个性质,即中序遍历二叉搜索树,前一个值必然小于后一个值。
代码实现
方案一:
class Solution {
public:
bool helper(TreeNode* root, long long lower, long long upper) {
if ( root == nullptr ) return true;
if ( root->val >= upper || root->val <= lower ) return false;
else return helper(root->left, lower, root->val) && helper(root->right, root->val, upper);
}
bool isValidBST(TreeNode* root) {
return helper(root, LONG_MIN, LONG_MAX);
}
};
方案2:
这里需要注意的是为了防止结点中有INT_MIN,我们用long long去存储最初的最小值。
class Solution {
public:
bool isValidBST(TreeNode* root) {
stack<TreeNode*> s;
long long front = (long long)INT_MIN - 1;
while ( !s.empty() || root ) {
while ( root ) {
s.push(root);
root = root->left;
}
root = s.top();
s.pop();
if ( root->val <= front ) return false;
front = root->val;
root = root->right;
}
return true;
}
};
反思
树的知识总是和递归结合在一起,对这方面的知识还是很欠缺,不过树和图都是非常重要的数据结构了,一定要勤加练习,共勉之。