栈的压入与弹出序列校验

简介: 栈的压入与弹出序列校验

前言


有两个整数序列,第一个序列表示栈的压入顺序,判断第二个序列是否为该栈的弹出顺序。假设压入栈的数字均不相等。例如,序列[1, 2, 3, 4, 5]是某栈的压栈序列,序列[4, 5, 3, 2, 1]是该栈序列对应的一个弹出序列,但[4, 3, 5, 1, 2]就不可能是该压栈序列的弹出序列。


思路分析


仔细分析题目后,我们很直观的想法就是构造一个辅助栈,把压入序列中的数字依次压入该辅助栈。按照弹出序列的顺序依次从该栈中弹出数字,如果辅助栈被清空则代表此序列是它的一个弹出序列,否则就不可能是一个弹出序列。


弹出序列满足条件


如下图所示,它的压入过程为:


  • 取出弹出序列的第1个元素,维护一个已取索引,在压入序列中从已取索引位置开始寻找与之相等的元素,将它之前的数字和其本身依次入栈,每取1个元素就将索引自增1次
  • 此时,栈顶元素与弹出序列的第1个元素相等,将栈顶元素出栈。
  • 取出弹出序列的第2个元素,在压入序列中从已取索引位置开始寻找与之相等的元素,将它之前的数字和其本身依次入栈。
  • 此时,栈顶元素与弹出序列的第2个元素相等,将栈顶元素出栈。
  • 取出弹出序列的第3个元素,此时,压入序列的元素已经被取完。我们继续判断 辅助栈中的元素是否与弹出序列的元素相等
  • 栈顶元素为3,要弹出的元素也是3,二者相等,栈顶元素出栈
  • 取出弹出序列的第4个元素
  • 栈顶元素为2,要弹出的元素也是2,二者相等,栈顶元素出栈
  • 取出弹出序列的第5个元素
  • 栈顶元素为1,要弹出的元素也是1,二者相等,栈顶元素出栈

弹出序列已取完,辅助栈已清空。 该弹出序列属于压入序列的一个弹出顺序


640.png

                           image-20220914102128919


弹出序列不满足条件


接下来,我们来分析下它不是压入序列的弹出顺序的情况,它的压入过程与满足条件时一样,唯独不同的是,弹出序列的第3个元素从辅助栈出栈后,压入序列已经被取完。此时,弹出序列的第4个元素是1,辅助栈的栈顶元素是2,二者不等,那么该序列肯定不是压入序列的弹出顺序。


640.png

                                    image-20220914114538508


实现代码


经过上面的分析,我们已经知道了如何解决这个问题。思路已明确,接下来,我们就可以愉快的进入编码环节了🤗


export function StackPushAndPopSequence(
  pushSequence: Array<number>,
  popupSequence: Array<number>
): boolean {
  if (pushSequence.length === 0 || popupSequence.length === 0) return false;
  // 下一个入栈、出栈索引
  let nextPushIndex = 0;
  let nextPopIndex = 0;
  // 辅助栈
  const stackData = new Stack();
  // 下一个弹出序列存在则执行进一步的判断
  while (nextPopIndex < popupSequence.length) {
    // 下一个弹出序列的元素与栈顶元素不等则入栈
    while (
      nextPushIndex < pushSequence.length &&
      popupSequence[nextPopIndex] !== stackData.peek()
    ) {
      stackData.push(pushSequence[nextPushIndex]);
      nextPushIndex++;
    }
    // 栈顶元素与下一个弹出序列元素相等则出栈
    if (stackData.peek() === popupSequence[nextPopIndex]) {
      stackData.pop();
      nextPopIndex++;
    } else {
      // 元素不等则终止循环,此时压入序列已经全部压入辅助栈,该序列不可能是一个弹出序列
      break;
    }
    // 辅助栈清空,则代表弹出序列是正确的
    if (stackData.isEmpty()) {
      return true;
    }
  }
  return false;
}


最后,我们将开头列举的例子来验证下上述代码是否正确执行,如下所示:


const pushSuite = [1, 2, 3, 4, 5];
const popSuite1 = [4, 5, 3, 2, 1];
const popSuite2 = [4, 3, 5, 1, 2];
const result1 = StackPushAndPopSequence(pushSuite, popSuite1);
const result2 = StackPushAndPopSequence(pushSuite, popSuite2);
console.log(result1, result2);


640.png

                                image-20220914203514154


示例代码


本文所用代码完整版请移步:


  • StackPushAndPopSequence.ts
  • stackPushAndPopSequence-test.ts


写在最后


至此,文章就分享完毕了。


我是神奇的程序员,一位前端开发工程师。


如果你对我感兴趣,请移步我的个人网站,进一步了解。

  • 公众号无法外链,如果文中有链接,可点击下方阅读原文查看😊
相关文章
|
19天前
|
消息中间件 存储 搜索推荐
深入理解栈和队列(二):队列
深入理解栈和队列(二):队列
34 0
|
1月前
【栈】数据结构栈的实现
【栈】数据结构栈的实现
|
1月前
|
存储
数据结构--栈和队列
数据结构--栈和队列
|
1月前
|
存储 算法 数据处理
数据结构从入门到精通——栈
栈,作为一种后进先出(LIFO)的数据结构,在计算机科学中扮演着重要的角色。它的特性使得它在处理函数调用、括号匹配、表达式求值等问题时具有得天独厚的优势。然而,如果我们跳出传统思维的束缚,会发现栈的用途远不止于此。
58 0
|
1月前
|
C语言
数据结构之栈详解(C语言手撕)
数据结构之栈详解(C语言手撕)
37 1
|
1月前
|
存储 算法
数据结构— —栈的基本操作(顺序栈和链栈)
数据结构— —栈的基本操作(顺序栈和链栈)
58 0
|
2天前
|
C语言
数据结构中顺序栈的进栈和出栈用C语言表示
数据结构中顺序栈的进栈和出栈用C语言表示
11 1
|
12天前
|
存储 算法 调度
数据结构期末复习(3)栈和队列
数据结构期末复习(3)栈和队列
18 0
|
24天前
|
存储 缓存 算法
【算法与数据结构】栈的实现详解
【算法与数据结构】栈的实现详解