返回介绍

solution / 0200-0299 / 0225.Implement Stack using Queues / README

发布于 2024-06-17 01:04:03 字数 6776 浏览 0 评论 0 收藏 0

225. 用队列实现栈

English Version

题目描述

请你仅使用两个队列实现一个后入先出(LIFO)的栈,并支持普通栈的全部四种操作(pushtoppopempty)。

实现 MyStack 类:

  • void push(int x) 将元素 x 压入栈顶。
  • int pop() 移除并返回栈顶元素。
  • int top() 返回栈顶元素。
  • boolean empty() 如果栈是空的,返回 true ;否则,返回 false

 

注意:

  • 你只能使用队列的标准操作 —— 也就是 push to backpeek/pop from frontsize 和 is empty 这些操作。
  • 你所使用的语言也许不支持队列。 你可以使用 list (列表)或者 deque(双端队列)来模拟一个队列 , 只要是标准的队列操作即可。

 

示例:

输入:
["MyStack", "push", "push", "top", "pop", "empty"]
[[], [1], [2], [], [], []]
输出:
[null, null, null, 2, 2, false]

解释:
MyStack myStack = new MyStack();
myStack.push(1);
myStack.push(2);
myStack.top(); // 返回 2
myStack.pop(); // 返回 2
myStack.empty(); // 返回 False

 

提示:

  • 1 <= x <= 9
  • 最多调用100pushpoptopempty
  • 每次调用 poptop 都保证栈不为空

 

进阶:你能否仅用一个队列来实现栈。

解法

方法一:两个队列

我们使用两个队列 $q_1$ 和 $q_2$,其中 $q_1$ 用于存储栈中的元素,而 $q_2$ 用于辅助实现栈的操作。

  • push 操作:将元素压入 $q_2$,然后将 $q_1$ 中的元素依次弹出并压入 $q_2$,最后交换 $q_1$ 和 $q_2$ 的引用。时间复杂度 $O(n)$。
  • pop 操作:直接弹出 $q_1$ 的队首元素。时间复杂度 $O(1)$。
  • top 操作:直接返回 $q_1$ 的队首元素。时间复杂度 $O(1)$。
  • empty 操作:判断 $q_1$ 是否为空。时间复杂度 $O(1)$。

空间复杂度 $O(n)$,其中 $n$ 是栈中元素的个数。

class MyStack:
  def __init__(self):
    self.q1 = deque()
    self.q2 = deque()

  def push(self, x: int) -> None:
    self.q2.append(x)
    while self.q1:
      self.q2.append(self.q1.popleft())
    self.q1, self.q2 = self.q2, self.q1

  def pop(self) -> int:
    return self.q1.popleft()

  def top(self) -> int:
    return self.q1[0]

  def empty(self) -> bool:
    return len(self.q1) == 0


# Your MyStack object will be instantiated and called as such:
# obj = MyStack()
# obj.push(x)
# param_2 = obj.pop()
# param_3 = obj.top()
# param_4 = obj.empty()
import java.util.Deque;

class MyStack {
  private Deque<Integer> q1 = new ArrayDeque<>();
  private Deque<Integer> q2 = new ArrayDeque<>();

  public MyStack() {
  }

  public void push(int x) {
    q2.offer(x);
    while (!q1.isEmpty()) {
      q2.offer(q1.poll());
    }
    Deque<Integer> q = q1;
    q1 = q2;
    q2 = q;
  }

  public int pop() {
    return q1.poll();
  }

  public int top() {
    return q1.peek();
  }

  public boolean empty() {
    return q1.isEmpty();
  }
}

/**
 * Your MyStack object will be instantiated and called as such:
 * MyStack obj = new MyStack();
 * obj.push(x);
 * int param_2 = obj.pop();
 * int param_3 = obj.top();
 * boolean param_4 = obj.empty();
 */
class MyStack {
public:
  MyStack() {
  }

  void push(int x) {
    q2.push(x);
    while (!q1.empty()) {
      q2.push(q1.front());
      q1.pop();
    }
    swap(q1, q2);
  }

  int pop() {
    int x = q1.front();
    q1.pop();
    return x;
  }

  int top() {
    return q1.front();
  }

  bool empty() {
    return q1.empty();
  }

private:
  queue<int> q1;
  queue<int> q2;
};

/**
 * Your MyStack object will be instantiated and called as such:
 * MyStack* obj = new MyStack();
 * obj->push(x);
 * int param_2 = obj->pop();
 * int param_3 = obj->top();
 * bool param_4 = obj->empty();
 */
type MyStack struct {
  q1 []int
  q2 []int
}

func Constructor() MyStack {
  return MyStack{}
}

func (this *MyStack) Push(x int) {
  this.q2 = append(this.q2, x)
  for len(this.q1) > 0 {
    this.q2 = append(this.q2, this.q1[0])
    this.q1 = this.q1[1:]
  }
  this.q1, this.q2 = this.q2, this.q1
}

func (this *MyStack) Pop() int {
  x := this.q1[0]
  this.q1 = this.q1[1:]
  return x
}

func (this *MyStack) Top() int {
  return this.q1[0]
}

func (this *MyStack) Empty() bool {
  return len(this.q1) == 0
}

/**
 * Your MyStack object will be instantiated and called as such:
 * obj := Constructor();
 * obj.Push(x);
 * param_2 := obj.Pop();
 * param_3 := obj.Top();
 * param_4 := obj.Empty();
 */
class MyStack {
  q1: number[] = [];
  q2: number[] = [];

  constructor() {}

  push(x: number): void {
    this.q2.push(x);
    while (this.q1.length) {
      this.q2.push(this.q1.shift()!);
    }
    [this.q1, this.q2] = [this.q2, this.q1];
  }

  pop(): number {
    return this.q1.shift()!;
  }

  top(): number {
    return this.q1[0];
  }

  empty(): boolean {
    return this.q1.length === 0;
  }
}

/**
 * Your MyStack object will be instantiated and called as such:
 * var obj = new MyStack()
 * obj.push(x)
 * var param_2 = obj.pop()
 * var param_3 = obj.top()
 * var param_4 = obj.empty()
 */
use std::collections::VecDeque;

struct MyStack {
  /// There could only be two status at all time
  /// 1. One contains N elements, the other is empty
  /// 2. One contains N - 1 elements, the other contains exactly 1 element
  q_1: VecDeque<i32>,
  q_2: VecDeque<i32>,
  // Either 1 or 2, originally begins from 1
  index: i32,
}

impl MyStack {
  fn new() -> Self {
    Self {
      q_1: VecDeque::new(),
      q_2: VecDeque::new(),
      index: 1,
    }
  }

  fn move_data(&mut self) {
    // Always move from q1 to q2
    assert!(self.q_2.len() == 1);
    while !self.q_1.is_empty() {
      self.q_2.push_back(self.q_1.pop_front().unwrap());
    }
    let tmp = self.q_1.clone();
    self.q_1 = self.q_2.clone();
    self.q_2 = tmp;
  }

  fn push(&mut self, x: i32) {
    self.q_2.push_back(x);
    self.move_data();
  }

  fn pop(&mut self) -> i32 {
    self.q_1.pop_front().unwrap()
  }

  fn top(&mut self) -> i32 {
    *self.q_1.front().unwrap()
  }

  fn empty(&self) -> bool {
    self.q_1.is_empty()
  }
}

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据
    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文