队列的典型题:栈实现队列、双端队列

简介: 队列的典型题:栈实现队列、双端队列

队列的典型题:栈实现队列、双端队列


最近在看数据结构和算法,努力总结出道~

TL;DR

队列的特点:先进先出。

练习:用栈实现队列

网络异常,图片无法展示
|

栈后进先出,队列先进先出。

思路是:「输入栈」正常push的时候,相当于队列的倒序;如果把「输入栈」的元素逐个弹出放到「输出栈」,相当于正序的队列,此时弹出就实现了先进先出。

步骤:

  • 把一个栈当做「输入栈」,把另一个栈当做「输出栈」
  • 当 push() 新元素的时候,进入「输入栈」
  • 当 pop() 元素的时候,优先从「输出栈」弹出元素。如果「输出栈」为空,则把「输入栈」的元素逐个推进到「输出栈」中,这样「输出栈」相当于正序的队列,可以执行pop的操作了

网络异常,图片无法展示
|

/**
 * Initialize your data structure here.
 */
var MyQueue = function () {
  //  取出只能从这个栈里,如果这个栈空的话,就把inStack都放进来,再取出栈顶
  this.outStack = [];
  //  进队只能从这个栈里
  this.inStack = [];
};
/**
 * Push element x to the back of queue.
 * @param {number} x
 * @return {void}
 */
MyQueue.prototype.push = function (x) {
  this.inStack.push(x);
};
/**
 * Removes the element from in front of queue and returns that element.
 * @return {number}
 */
MyQueue.prototype.pop = function () {
  if (this.outStack.length) {
    return this.outStack.pop();
  }
  while (this.inStack.length) {
    this.outStack.push(this.inStack.pop());
  }
  return this.outStack.pop();
};
/**
 * Get the front element.
 * @return {number}
 */
MyQueue.prototype.peek = function () {
  if (this.outStack.length) {
    return this.outStack[this.outStack.length - 1];
  }
  while (this.inStack.length) {
    this.outStack.push(this.inStack.pop());
  }
  return this.outStack.length && this.outStack[this.outStack.length - 1];
};
/**
 * Returns whether the queue is empty.
 * @return {boolean}
 */
MyQueue.prototype.empty = function () {
  return !this.outStack.length && !this.inStack.length;
};

这里注意,虽然取出的操作,可能有循环,但是均摊时间复杂度其实O(1),可以简单理解为大部分的取出都是O(1)。

官方的解法

练习:双端队列

双端队列就是允许在队列的两端进行插入和删除的队列。

体现在编码上,最常见的载体是既允许使用 pop、push 同时又允许使用 shift、unshift 的数组。

网络异常,图片无法展示
|

普通办法:循环,找出每个窗口的最大值,时间复杂度O(kn)。

于是,还是用空间换时间,采用双向队列法,让复杂度变成O(n)

思路:窗口移动,维护队列,让队首始终是当前窗口的最大值的索引。队列是递减队列,存放索引,推进值之后,判断队首是不是在窗口之外,之外则移除。移除之后,新的队首就是当前窗口的最大值。

步骤:

  • 遍历给定数组中的元素,
  • 如果队列不为空且当前值 > 队尾元素,则将队尾元素移除。直到,队列为空或当前值小于新的队尾元素,才将当前值推进队列;
  • 当队首元素的下标 < 当前窗口左边界L时,表示队首元素已不在滑动窗口内,需要将其从队首移除。
  • 此时,队首元素就是该窗口内的最大值。但由于数组下标从0开始,只有当窗口右边界R>=k-1时,才意味着窗口形成,注意这个前提条件。

网络异常,图片无法展示
|

演示和思路是元素,但实际代码中,为了方便操作,队列存储的是索引。

const last = (arr) => arr[arr.length - 1];
const first = (arr) => arr[0];
var maxSlidingWindow = function (nums, k) {
  // 存储最大值的数组
  let res = [];
  // 递减队列
  let queue = [];
  // 遍历,R表示滑动窗口右边界
  for (let R = 0; R < nums.length; R++) {
    const cur = nums[R];
    // 维护递减队列,只要当前值大于队列末端,就pop,直到小于或者为空才进队
    while (queue.length && cur > nums[last(queue)]) {
      queue.pop();
    }
    queue.push(R);
    // 窗口左边的指针索引
    const L = R - k + 1;
    // 队首的索引小于L,表示在窗口之外了,需要移除
    if (first(queue) < L) {
      queue.shift();
    }
    // 当R>=k-1的时候,窗口形成,此时队首是最大值的索引,存到res中
    if (R >= k - 1) {
      res[L] = nums[first(queue)];
    }
  }
  return res;
};

详细图解和其他官方其他方案

引用

目录
相关文章
|
10月前
|
前端开发 Java
java实现队列数据结构代码详解
本文详细解析了Java中队列数据结构的实现,包括队列的基本概念、应用场景及代码实现。队列是一种遵循“先进先出”原则的线性结构,支持在队尾插入和队头删除操作。文章介绍了顺序队列与链式队列,并重点分析了循环队列的实现方式以解决溢出问题。通过具体代码示例(如`enqueue`入队和`dequeue`出队),展示了队列的操作逻辑,帮助读者深入理解其工作机制。
397 1
|
8月前
|
编译器 C语言 C++
栈区的非法访问导致的死循环(x64)
这段内容主要分析了一段C语言代码在VS2022中形成死循环的原因,涉及栈区内存布局和数组越界问题。代码中`arr[15]`越界访问,修改了变量`i`的值,导致`for`循环条件始终为真,形成死循环。原因是VS2022栈区从低地址到高地址分配内存,`arr`数组与`i`相邻,`arr[15]`恰好覆盖`i`的地址。而在VS2019中,栈区先分配高地址再分配低地址,因此相同代码表现不同。这说明编译器对栈区内存分配顺序的实现差异会导致程序行为不一致,需避免数组越界以确保代码健壮性。
192 0
栈区的非法访问导致的死循环(x64)
|
存储 C语言 C++
【C++数据结构——栈与队列】顺序栈的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现顺序栈的基本运算。开始你的任务吧,祝你成功!​ 相关知识 初始化栈 销毁栈 判断栈是否为空 进栈 出栈 取栈顶元素 1.初始化栈 概念:初始化栈是为栈的使用做准备,包括分配内存空间(如果是动态分配)和设置栈的初始状态。栈有顺序栈和链式栈两种常见形式。对于顺序栈,通常需要定义一个数组来存储栈元素,并设置一个变量来记录栈顶位置;对于链式栈,需要定义节点结构,包含数据域和指针域,同时初始化栈顶指针。 示例(顺序栈): 以下是一个简单的顺序栈初始化示例,假设用C语言实现,栈中存储
688 77
232.用栈实现队列,225. 用队列实现栈
在232题中,通过两个栈(`stIn`和`stOut`)模拟队列的先入先出(FIFO)行为。`push`操作将元素压入`stIn`,`pop`和`peek`操作则通过将`stIn`的元素转移到`stOut`来实现队列的顺序访问。 225题则是利用单个队列(`que`)模拟栈的后入先出(LIFO)特性。通过多次调整队列头部元素的位置,确保弹出顺序符合栈的要求。`top`操作直接返回队列尾部元素,`empty`判断队列是否为空。 两题均仅使用基础数据结构操作,展示了栈与队列之间的转换逻辑。
|
12月前
|
算法 调度 C++
STL——栈和队列和优先队列
通过以上对栈、队列和优先队列的详细解释和示例,希望能帮助读者更好地理解和应用这些重要的数据结构。
301 11
|
存储 C++ 索引
【C++数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】
【数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】初始化队列、销毁队列、判断队列是否为空、进队列、出队列等。本关任务:编写一个程序实现环形队列的基本运算。(6)出队列序列:yzopq2*(5)依次进队列元素:opq2*(6)出队列序列:bcdef。(2)依次进队列元素:abc。(5)依次进队列元素:def。(2)依次进队列元素:xyz。开始你的任务吧,祝你成功!(4)出队一个元素a。(4)出队一个元素x。
495 13
【C++数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】
☀☀☀☀☀☀☀有关栈和队列应用的oj题讲解☼☼☼☼☼☼☼
### 简介 本文介绍了三种数据结构的实现方法:用两个队列实现栈、用两个栈实现队列以及设计循环队列。具体思路如下: 1. **用两个队列实现栈**: - 插入元素时,选择非空队列进行插入。 - 移除栈顶元素时,将非空队列中的元素依次转移到另一个队列,直到只剩下一个元素,然后弹出该元素。 - 判空条件为两个队列均为空。 2. **用两个栈实现队列**: - 插入元素时,选择非空栈进行插入。 - 移除队首元素时,将非空栈中的元素依次转移到另一个栈,再将这些元素重新放回原栈以保持顺序。 - 判空条件为两个栈均为空。
|
C++
【C++数据结构——栈和队列】括号配对(头歌实践教学平台习题)【合集】
【数据结构——栈和队列】括号配对(头歌实践教学平台习题)【合集】(1)遇到左括号:进栈Push()(2)遇到右括号:若栈顶元素为左括号,则出栈Pop();否则返回false。(3)当遍历表达式结束,且栈为空时,则返回true,否则返回false。本关任务:编写一个程序利用栈判断左、右圆括号是否配对。为了完成本关任务,你需要掌握:栈对括号的处理。(1)遇到左括号:进栈Push()开始你的任务吧,祝你成功!测试输入:(()))
349 7
|
C语言
【数据结构】栈和队列(c语言实现)(附源码)
本文介绍了栈和队列两种数据结构。栈是一种只能在一端进行插入和删除操作的线性表,遵循“先进后出”原则;队列则在一端插入、另一端删除,遵循“先进先出”原则。文章详细讲解了栈和队列的结构定义、方法声明及实现,并提供了完整的代码示例。栈和队列在实际应用中非常广泛,如二叉树的层序遍历和快速排序的非递归实现等。
1140 9
|
存储 算法
非递归实现后序遍历时,如何避免栈溢出?
后序遍历的递归实现和非递归实现各有优缺点,在实际应用中需要根据具体的问题需求、二叉树的特点以及性能和空间的限制等因素来选择合适的实现方式。
359 59