找回密码
 立即注册
首页 业界区 业界 剑指offer-23、搜索⼆叉树的后序遍历序列 ...

剑指offer-23、搜索⼆叉树的后序遍历序列

刘凤 昨天 19:05
题⽬描述

输⼊⼀个整数数组,判断该数组是不是某⼆叉搜索树的后序遍历的结果。如果是则返回true,否则返回false 。假设输⼊的数组的任意两个数字都互不相同。
提示:

  • ⼆叉搜索树是指⽗亲节点⼤于左⼦树中的全部节点,但是⼩于右⼦树中的全部节点的树。
  • 该题我们约定空树不是⼆叉搜索树
  • 后序遍历是指按照 “左⼦树-右⼦树-根节点” 的顺序遍历
  • 参考下⾯的⼆叉搜索树,示例 1
示例1
1.jpeg

输⼊:[1,3,2]
返回值:true
说明:是上图的后序遍历 ,返回true
思路及解答

需要判断给定的整数数组是否是某个二叉搜索树(BST)的后序遍历结果。二叉搜索树具有以下特性:

  • 左子树所有节点的值小于根节点的值
  • 右子树所有节点的值大于根节点的值
  • 左右子树也必须是二叉搜索树
后序遍历的顺序是:左子树 → 右子树 → 根节点,因此数组的最后一个元素一定是根节点
递归(标准解法)

注意是⼆叉搜索树,如果是后续遍历的话,那么应该最后⼀个元素是中间元素 mid ,前⾯的元素可以分为两部分,⼀部分⽐ mid ⼩,另⼀部分全部⽐ mid ⼤。如果破坏这个原则,那么就应该输出No 。采取分⽽治之的⽅法,分别递归检查左⼦树以及右⼦树:

  • 确定根节点​:后序遍历的最后一个元素是根节点
  • 划分左右子树​:从数组开头找到第一个大于根节点的元素,该元素之前为左子树,之后到根节点前为右子树
  • 验证BST性质​:右子树所有元素必须大于根节点
  • 递归验证​:对左右子树重复上述过程
  1. public class Solution {
  2.     public boolean VerifySquenceOfBST(int[] postorder) {
  3.         if (postorder == null || postorder.length == 0) {
  4.             return false; // 空树不是BST
  5.         }
  6.         return verify(postorder, 0, postorder.length - 1);
  7.     }
  8.    
  9.     private boolean verify(int[] postorder, int start, int end) {
  10.         if (start >= end) {
  11.             return true; // 单个节点总是BST
  12.         }
  13.         
  14.         int root = postorder[end]; // 根节点是最后一个元素
  15.         int i = start;
  16.         // 找到第一个大于根节点的元素,作为左右子树分界
  17.         while (i < end && postorder[i] < root) {
  18.             i++;
  19.         }
  20.         
  21.         // 检查右子树是否都大于根节点
  22.         for (int j = i; j < end; j++) {
  23.             if (postorder[j] < root) {
  24.                 return false;
  25.             }
  26.         }
  27.         
  28.         // 递归检查左右子树
  29.         return verify(postorder, start, i - 1) && verify(postorder, i, end - 1);
  30.     }
  31. }
复制代码

  • 时间复杂度:O($n^2$),n 为⼆叉树节点的个数,当树为链式时时间复杂度最坏为 O($n^2$)
  • 空间复杂度:O(n),当树为链式结构时,递归深度为 n
单调栈法

利用单调栈和后序遍历的倒序特性:

  • 逆序处理​:后序遍历的逆序是"根→右→左",类似于变种的前序遍历
  • 维护最小值​:使用单调栈保持递增顺序,遇到较小值时弹出并更新最小值
  • 验证性质​:确保当前元素不大于最小值(即违反BST性质)
2.png

3.png

4.png

5.png

6.png

7.png

8.png

9.png

10.png

11.png
  1. public boolean verifyPostorder(int[] postorder) {
  2.     if (postorder == null || postorder.length == 0) {
  3.         return false;
  4.     }
  5.    
  6.     Stack<Integer> stack = new Stack<>();
  7.     int max = Integer.MAX_VALUE; // 初始化最大值
  8.    
  9.     // 从后往前遍历
  10.     for (int i = postorder.length - 1; i >= 0; i--) {
  11.         int current = postorder[i];
  12.         // 如果当前节点大于max,违反BST性质
  13.         if (current > max) {
  14.             return false;
  15.         }
  16.         
  17.         // 弹出所有比当前节点大的元素,并更新max
  18.         while (!stack.isEmpty() && stack.peek() > current) {
  19.             max = stack.pop();
  20.         }
  21.         
  22.         stack.push(current);
  23.     }
  24.    
  25.     return true;
  26. }
复制代码

  • 时间复杂度​:O(n),每个元素最多入栈和出栈一次
  • 空间复杂度​:O(n),最坏情况下需要存储所有元素

来源:程序园用户自行投稿发布,如果侵权,请联系站长删除
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
您需要登录后才可以回帖 登录 | 立即注册