声明:问题描述来源于leetcode
一、问题描述:
用队列实现栈
Category | Difficulty | Likes | Dislikes |
---|
algorithms | Easy (67.64%) | 480 | - |
Tags
stack
| design
Companies
bloomberg
请你仅使用两个队列实现一个后入先出(LIFO)的栈,并支持普通栈的全部四种操作(push
、top
、pop
和 empty
)。
实现 MyStack
类:
void push(int x)
将元素 x 压入栈顶。int pop()
移除并返回栈顶元素。int top()
返回栈顶元素。boolean empty()
如果栈是空的,返回 true
;否则,返回 false
。
注意:
- 你只能使用队列的基本操作 —— 也就是
push to back
、peek/pop from front
、size
和 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
- 最多调用
100
次 push
、pop
、top
和 empty
- 每次调用
pop
和 top
都保证栈不为空
**进阶:**你能否仅用一个队列来实现栈。
Discussion | Solution
二、题解1:通过链表实现堆栈
通过list作为容器,直接见招拆招,要什么方法就通过list来实现什么方法,xin麒觉得应该还是比较容易懂的。
xin麒的思路:
- 首先定义List类型的成员变量list
- 1、构造方法:初始化list
- void push(int x):通过分析堆栈的push方法的特点:将元素添加到末尾
- int pop():因为堆栈的push方法是移除并返回栈顶元素
- 于是直接通过list的remove(int index)方法传入末元素的索引即可提取所要求的元素
- int top():和pop方法类似,不过换成list的get(int index)方法即可
- boolean empty():通过三目运算直接返回
list.size() == 0
的运行结果即可
class MyStack {
List<Integer> list;
public MyStack() {
list = new LinkedList<>();
}
public void push(int x) {
list.add(x);
}
public int pop() {
if (list.size() != 0){
return list.remove(list.size() - 1);
}
return 0;
}
public int top() {
if (list.size() != 0){
return list.get(list.size() - 1);
}
return 0;
}
public boolean empty() {
return list.size() == 0;
}
}
运行:
三、题解2:通过两个队列实现堆栈
思考过程:
要通过两个队列实现堆栈,那么就要比较队列和堆栈的区别:
-
共同点:push方法是比较类似的比较好办,直接使用add()方法传入参数即可
empty()方法也是比较容易的,因为直接判断队列是否为空即可
-
不同点:pop方法中在堆栈的功能是:移除并返回栈顶元素;
在队列的功能是:移除并返回队列首元素。
(1)、int pop():
top方法和pop方法类似,于是难点便是如何处理pop方法了:
那么如果将第一个队列的元素一个一个往第二个队列放会有什么效果呢?
在这个过程的细节是:第一个队列的首元素不断通过pop方法添加到第二个队列里面,其中第二个队列是使用push来接收。
于是xin麒想到可以在第一个队列不断调用pop方法时,等到第一个队列只剩下1个元素时,不就是所需提取的元素了吗?因此问题就基本解决了,当第一个队列的剩下1个元素时,直接提取该元素,后面再将这个元素返回即可。
然后为了配合push方法的使用(因为此时如果再给堆栈添加元素,那么就要添加到第二个队列里面),于是将第一个队列指向第二个队列,第二个队列就重开吧。
(2)、boolean empty():
于是可以把第一个队列称为主队列mainQueue
,第二个队列称为副本队列subordinateQueue
。因此empty方法直接判断主队列是否为空即可。
(3)、int top():
top方法和pop方法比较类似,把提取的元素添加追加到副本队列即可。
class MyStack {
Queue<Integer> mainQueue;
Queue<Integer> subordinateQueue;
public MyStack() {
mainQueue = new LinkedList<>();
subordinateQueue = new LinkedList<>();
}
public void push(int x) {
mainQueue.add(x);
}
public int pop() {
if (!mainQueue.isEmpty()){
while (mainQueue.size() != 1){
subordinateQueue.add(mainQueue.poll());
}
int result = mainQueue.poll();
mainQueue = subordinateQueue;
subordinateQueue = new LinkedList<>();
return result;
}
return 0;
}
public int top() {
if (!mainQueue.isEmpty()) {
while (mainQueue.size() != 1) {
subordinateQueue.add(mainQueue.poll());
}
int result = mainQueue.peek();
subordinateQueue.add(result);
mainQueue = subordinateQueue;
subordinateQueue = new LinkedList<>();
return result;
}
return 0;
}
public boolean empty() {
return mainQueue.isEmpty();
}
}
运行:
简化:
因为有些代码可以复合的,xin麒就把它们封装在一个函数了,这样看起来比较简洁些。
class MyStack {
Queue<Integer> mainQueue;
Queue<Integer> subordinateQueue;
public MyStack() {
mainQueue = new LinkedList<>();
subordinateQueue = new LinkedList<>();
}
public void push(int x) {
mainQueue.add(x);
}
public int xinDo(){
int result = 0;
if (!mainQueue.isEmpty()){
while (mainQueue.size() != 1){
subordinateQueue.add(mainQueue.poll());
}
result = mainQueue.poll();
mainQueue = subordinateQueue;
subordinateQueue = new LinkedList<>();
}
return result;
}
public int pop() {
return xinDo();
}
public int top() {
int result = xinDo();
mainQueue.add(result);
return result;
}
public boolean empty() {
return mainQueue.isEmpty();
}
}
四、题解3:一个队列实现堆栈:
思路:
1、push方法不变,依然是使用list的add方法;
2(重点)、不断将队列首元素往末尾元素追加,同时移除该首元素,直到首元素为栈顶元素时便提取该元素作为返回值。
pop方法和top方法类似,稍做处理即可;
empty方法判断队列是否为空即可。
int pop()
:
思考过程:
- 不断将队列首元素往尾部扔的过程中如何知道轮到栈顶元素了呢?这个问题很重要,因此可以通过size()得到变量size来处理扔的过程,使用变量result来保存要返回的结果。
- 如果queue为非空,那么将队列的首元素扔出来(poll),同时扔进(add)该队列的尾部,在一直扔的过程中,
(--size >= 1)
作为循环退出的条件。当条件为false时退出循环,并且此时的首元素即为所要返回的元素,于是在队列里面移除该元素,存放到result里面。
int top
:
- 如果队列非空,那么这里可以直接调用pop()并且以result接收,因为pop方法已经将队列的元素删除了,于是再add回来即可。
class MyStack {
Queue<Integer> queue;
public MyStack() {
queue = new LinkedList<>();
}
public void push(int x) {
queue.add(x);
}
public int pop() {
int size = queue.size();
int result = 0;
if (size != 0){
while (--size >= 1){
queue.add(queue.poll());
}
result = queue.poll();
}
return result;
}
public int top() {
if (queue.isEmpty()){
return 0;
}
int result = pop();
queue.add(result);
return result;
}
public boolean empty() {
return queue.isEmpty();
}
}
运行:
end
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)