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

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

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


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

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;
};

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

引用

目录
相关文章
|
18天前
|
C语言
【数据结构】栈和队列(c语言实现)(附源码)
本文介绍了栈和队列两种数据结构。栈是一种只能在一端进行插入和删除操作的线性表,遵循“先进后出”原则;队列则在一端插入、另一端删除,遵循“先进先出”原则。文章详细讲解了栈和队列的结构定义、方法声明及实现,并提供了完整的代码示例。栈和队列在实际应用中非常广泛,如二叉树的层序遍历和快速排序的非递归实现等。
94 9
|
9天前
|
存储 算法
非递归实现后序遍历时,如何避免栈溢出?
后序遍历的递归实现和非递归实现各有优缺点,在实际应用中需要根据具体的问题需求、二叉树的特点以及性能和空间的限制等因素来选择合适的实现方式。
19 1
|
12天前
|
存储 算法 Java
数据结构的栈
栈作为一种简单而高效的数据结构,在计算机科学和软件开发中有着广泛的应用。通过合理地使用栈,可以有效地解决许多与数据存储和操作相关的问题。
|
15天前
|
存储 JavaScript 前端开发
执行上下文和执行栈
执行上下文是JavaScript运行代码时的环境,每个执行上下文都有自己的变量对象、作用域链和this值。执行栈用于管理函数调用,每当调用一个函数,就会在栈中添加一个新的执行上下文。
|
17天前
|
存储
系统调用处理程序在内核栈中保存了哪些上下文信息?
【10月更文挑战第29天】系统调用处理程序在内核栈中保存的这些上下文信息对于保证系统调用的正确执行和用户程序的正常恢复至关重要。通过准确地保存和恢复这些信息,操作系统能够实现用户模式和内核模式之间的无缝切换,为用户程序提供稳定、可靠的系统服务。
44 4
|
21天前
|
算法 安全 NoSQL
2024重生之回溯数据结构与算法系列学习之栈和队列精题汇总(10)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构王道第3章之IKUN和I原达人之数据结构与算法系列学习栈与队列精题详解、数据结构、C++、排序算法、java、动态规划你个小黑子;这都学不会;能不能不要给我家鸽鸽丢脸啊~除了会黑我家鸽鸽还会干嘛?!!!
|
1月前
数据结构(栈与列队)
数据结构(栈与列队)
19 1
|
1月前
|
存储 JavaScript 前端开发
为什么基础数据类型存放在栈中,而引用数据类型存放在堆中?
为什么基础数据类型存放在栈中,而引用数据类型存放在堆中?
68 1
|
1月前
【数据结构】-- 栈和队列
【数据结构】-- 栈和队列
17 0
|
1月前
|
算法 程序员 索引
数据结构与算法学习七:栈、数组模拟栈、单链表模拟栈、栈应用实例 实现 综合计算器
栈的基本概念、应用场景以及如何使用数组和单链表模拟栈,并展示了如何利用栈和中缀表达式实现一个综合计算器。
31 1
数据结构与算法学习七:栈、数组模拟栈、单链表模拟栈、栈应用实例 实现 综合计算器

热门文章

最新文章