LeetCode 98验证二叉搜素树(中序遍历)&99恢复二叉搜索树

简介: 给定一个二叉树,判断其是否是一个有效的二叉搜索树。

验证二叉搜索树



给定一个二叉树,判断其是否是一个有效的二叉搜索树。


假设一个二叉搜索树具有如下特征:


  • 节点的左子树只包含小于当前节点的数。
  • 节点的右子树只包含大于当前节点的数。
  • 所有左子树和右子树自身必须也是二叉搜索树。


示例 1:


输入:
    2
   / \
  1   3
输出: true


示例 2:


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


分析


二叉搜索树的中序遍历是一个递增序列,所以我们只需要进行二叉树的中序遍历查询序列是否一直有序即可。当然,如果你直接利用二叉搜索树的定义左节点<根节点<右节点的话当然也没问题。具体实现上,你可以先将中序序列求出来判断是否有序,也可直接利用一个临时存储上一个节点的值进行判断。


具体代码为:


//  public boolean isValidBST(TreeNode root) {
//    List<Integer>list=new ArrayList<Integer>();
//    dfs(root,list);
//    if(list.size()<=1)return true;
//    for(int i=1;i<list.size();i++)
//    {
//      if(list.get(i)<=list.get(i-1))
//        return false;
//    }
//    return true;
//  }
//  private void dfs(TreeNode root, List<Integer> list) {
//    // TODO Auto-generated method stub
//    if(root==null)
//      return;
//    dfs(root.left, list);
//    list.add(root.val);
//    dfs(root.right, list);
//  }
long pre=Long.MIN_VALUE;
public boolean isValidBST(TreeNode root) {
   if(root==null)
     return true;
   if(!isValidBST(root.left))
     return false;
   if(pre>=root.val)
     return false;
   pre=root.val;
   if(!isValidBST(root.right))
     return false;
   return true; 
  } 


恢复二叉搜索树



给你二叉搜索树的根节点 root ,该树中的两个节点被错误地交换。请在不改变其结构的情况下,恢复这棵树。


进阶:使用 O(n) 空间复杂度的解法很容易实现。你能想出一个只使用常数空间的解决方案吗?


示例 1:


20210103151616215.png


输入:root = [1,3,null,null,2]
输出:[3,1,null,null,2]
解释:3 不能是 1 左孩子,因为 3 > 1 。交换 1 和 3 使二叉搜索树有效。


示例 2:


20210103151626868.png


输入:root = [3,1,4,null,null,2]
输出:[2,1,4,null,null,3]
解释:2 不能在 3 的右子树中,因为 2 < 3 。交换 2 和 3 使二叉搜索树有效。


提示:


树上节点的数目在范围 [2, 1000] 内
-2^31 <= Node.val <= 2^31 - 1


分析:


这题主要是需要动手画一画分析。首先它告诉本来一个二叉搜索树有两个节点的值被交换,我们知道对应一个正常的二叉搜索树它的中序排序序列是有序递增的,那么要将它们的数值进行交换。


分析一个序列1 3 5 7 8 9 10 假如其中的3和9进行交换次序(选择模拟的时候要么选择非常远要么选择非常更直观)那么有两段逆序,两个分别可以确定大小点。如果两个相邻交换那么仅有一段逆序对,分别是交换的两个节点。


当然在具体实现的时候你可以根据位置进行记录,一旦到达第二个逆序就可以停止。这题中序遍历上你可以直接先一次求中序序列,然后找到两个逆序,第二次再去进行交换(O(n))。当然更好的方法是一次就去把这个过程搞完,需要借助一个前驱节点去比较前一个数(O(1))。


20210103193922972.png


具体代码为:


//   public void recoverTree(TreeNode root) {
//     List<Integer>list=new ArrayList<Integer>();
//     dfs1(root,list);
//     int l=-1,lvalue=0,r=-1,rvalue=0;
//     for(int i=1;i<list.size();i++)
//     {
//       if(list.get(i)<list.get(i-1))
//       {
//         if(l==-1) {
//           l=i-1;
//           lvalue=list.get(i-1);
//         }
//         r=i;
//         rvalue=list.get(i); 
//       }
//     }
//     dfs2(root,l,lvalue,r,rvalue);
//   }
//   int index=0;
//  private void dfs2(TreeNode root, int l, int lvalue, int r, int rvalue) {
//    if(root==null)
//      return;
//    dfs2(root.left, l, lvalue, r, rvalue);
//    if(index>r)
//      return;
//    if(index==l)
//      root.val=rvalue;
//    else if(index==r)
//      root.val=lvalue;
//    index++;
//    dfs2(root.right, l, lvalue, r, rvalue); 
//  }
//  private void dfs1(TreeNode root, List<Integer> list) {
//    // TODO Auto-generated method stub
//    if(root==null)
//      return;
//    dfs1(root.left, list);
//    list.add(root.val);
//    dfs1(root.right, list);
//    
//  }
 public void recoverTree(TreeNode root) {
  Stack<TreeNode> q1 = new Stack(); 
  TreeNode first=null;
  TreeNode second=null;
  TreeNode pre=new TreeNode(Integer.MIN_VALUE);
  TreeNode t=root;
  while(!q1.isEmpty()||t!=null)
  {
    if (t!=null) {
      q1.push(t);
      t=t.left;
    }
    else {
      t=q1.pop();
      //中序操作
      if(t.val<pre.val)
      {
        if(first==null)
          first=pre;
        else {
          second=t;
          break;//可以停止了
        }
        second=t;
      }
      pre=t;//更新下pre
      t=t.right;
    }
  }
  int team=first.val;
  first.val=second.val;
  second.val=team;
 }


原创不易,bigsai请你帮两件事帮忙一下:


star支持一下, 您的肯定是我在平台创作的源源动力。


微信搜索「bigsai」,关注我的公众号,不仅免费送你电子书,我还会第一时间在公众号分享知识技术。加我还可拉你进力扣打卡群一起打卡LeetCode。


记得关注、咱们下次再见!


目录
相关文章
|
1月前
|
Python
【Leetcode刷题Python】剑指 Offer 26. 树的子结构
这篇文章提供了解决LeetCode上"剑指Offer 26. 树的子结构"问题的Python代码实现和解析,判断一棵树B是否是另一棵树A的子结构。
34 4
|
1月前
|
Python
【Leetcode刷题Python】450. 删除二叉搜索树中的节点
LeetCode上538号问题"把二叉搜索树转换为累加树"的Python实现,使用反向中序遍历并记录节点值之和来更新每个节点的新值。
25 4
【Leetcode刷题Python】450. 删除二叉搜索树中的节点
|
1月前
|
Python
【Leetcode刷题Python】96. 不同的二叉搜索树
LeetCode 96题 "不同的二叉搜索树" 的Python解决方案,使用动态规划算法计算由1至n互不相同节点值组成的二叉搜索树的种数。
14 3
【Leetcode刷题Python】96. 不同的二叉搜索树
|
1月前
|
算法
LeetCode第96题不同的二叉搜索树
文章介绍了LeetCode第96题"不同的二叉搜索树"的解法,利用动态规划的思想和递推公式,通过计算以任意节点为根的不同二叉搜索树的数量,解决了该问题。
LeetCode第96题不同的二叉搜索树
|
1月前
|
算法 Java
LeetCode第94题二叉树的中序遍历
文章介绍了LeetCode第94题"二叉树的中序遍历"的解法,使用递归实现了中序遍历的过程,遵循了"左根右"的遍历顺序,并提供了清晰的Java代码实现。
LeetCode第94题二叉树的中序遍历
|
1月前
|
算法 Python
【Leetcode刷题Python】剑指 Offer 33. 二叉搜索树的后序遍历序列
本文提供了一种Python算法,用以判断给定整数数组是否为某二叉搜索树的后序遍历结果,通过识别根节点并递归验证左右子树的值是否满足二叉搜索树的性质。
14 3
|
1月前
|
Python
【Leetcode刷题Python】538. 把二叉搜索树转换为累加树
LeetCode上538号问题"把二叉搜索树转换为累加树"的Python实现,使用反向中序遍历并记录节点值之和来更新每个节点的新值。
17 3
|
1月前
|
Python
【Leetcode刷题Python】108. 将有序数组转换为二叉搜索树
LeetCode上108号问题"将有序数组转换为二叉搜索树"的Python实现,通过递归选取数组中间值作为根节点,构建高度平衡的二叉搜索树。
21 2
|
1月前
|
Python
【Leetcode刷题Python】105. 从前序与中序遍历序列构造二叉树
LeetCode上105号问题"从前序与中序遍历序列构造二叉树"的Python实现,通过递归方法根据前序和中序遍历序列重建二叉树。
17 3
|
1月前
|
Python
【Leetcode刷题Python】145. 二叉树的后序遍历
LeetCode上145号问题"二叉树的后序遍历"的Python实现方法。
15 2