225. 用队列实现栈

目录

思路

方法一:一个队列模拟栈

方法二:用两个队列模拟栈

方法三:用队列实现栈(此方法并没有模拟队列的特性,先进先出)

 方法四:使用列表实现栈(此方法并没有模拟队列的特性,先进先出)

心得收获 

参考文章


225. 用队列实现栈

思路

队列模拟栈,其实一个队列就够了,那么我们先说一说两个队列来实现栈的思路。

队列是先进先出的规则,把一个队列中的数据导入另一个队列中,数据的顺序并没有变,并没有变成先进后出的顺序。

所以用栈实现队列, 和用队列实现栈的思路还是不一样的,这取决于这两个数据结构的性质。

但是依然还是要用两个队列来模拟栈,只不过没有输入和输出的关系,而是另一个队列完全用来备份的!

如下面动画所示,用两个队列que1和que2实现队列的功能,que2其实完全就是一个备份的作用,把que1最后面的元素以外的元素都备份到que2,然后弹出最后面的元素,再把其他元素从que2导回que1。

方法一:一个队列模拟栈

class MyStack:
    def __init__(self) -> None:
        self.stack = deque()

    # 元素 x 入栈
    def push(self,x:int) -> None:
        self.stack.append(x)

    # 移除栈顶元素
    def pop(self) -> None:
        if self.empty():
            return None
        else:
            for i in range(len(self.stack)-1):
                self.stack.append(self.stack.popleft())

            return self.stack.popleft()

    # 获取栈顶元素

    def top(self) -> None:
        # 写法一:
        # if self.empty():
        #     return None
        # return self.que[-1]
        
        # 写法二
        if self.empty():
            return None
        else:
            ans = self.stack.pop()
            self.stack.append(ans)
            return ans
    
    # 返回栈是否为空
    def empty(self) ->None:

        return not self.stack

方法二:用两个队列模拟栈

from collections import deque

class MyStack:

    def __init__(self):
        """
        Python普通的Queue或SimpleQueue没有类似于peek的功能
        也无法用索引访问,在实现top的时候较为困难。

        用list可以,但是在使用pop(0)的时候时间复杂度为O(n)
        因此这里使用双向队列,我们保证只执行popleft()和append(),因为deque可以用索引访问,可以实现和peek相似的功能

        in - 存所有数据
        out - 仅在pop的时候会用到
        """
        self.queue_in = deque()
        self.queue_out = deque()

    def push(self, x: int) -> None:
        """
        直接append即可
        """
        self.queue_in.append(x)


    def pop(self) -> int:
        """
        1. 首先确认不空
        2. 因为队列的特殊性,FIFO,所以我们只有在pop()的时候才会使用queue_out
        3. 先把queue_in中的所有元素(除了最后一个),依次出列放进queue_out
        4. 交换in和out,此时out里只有一个元素
        5. 把out中的pop出来,即是原队列的最后一个
        
        tip:这不能像栈实现队列一样,因为另一个queue也是FIFO,如果执行pop()它不能像
        stack一样从另一个pop(),所以干脆in只用来存数据,pop()的时候两个进行交换
        """
        if self.empty():
            return None

        for i in range(len(self.queue_in) - 1):
            self.queue_out.append(self.queue_in.popleft())
        
        self.queue_in, self.queue_out = self.queue_out, self.queue_in    # 交换in和out,这也是为啥in只用来存
        return self.queue_out.popleft()

    def top(self) -> int:
        """
        写法一:
        1. 首先确认不空
        2. 我们仅有in会存放数据,所以返回第一个即可(这里实际上用到了栈)
        写法二:
        1. 首先确认不空
        2. 因为队列的特殊性,FIFO,所以我们只有在pop()的时候才会使用queue_out
        3. 先把queue_in中的所有元素(除了最后一个),依次出列放进queue_out
        4. 交换in和out,此时out里只有一个元素
        5. 把out中的pop出来,即是原队列的最后一个,并使用temp变量暂存
        6. 把temp追加到queue_in的末尾
        """
        # 写法一:
        # if self.empty():
        #     return None
        
        # return self.queue_in[-1]    # 这里实际上用到了栈,因为直接获取了queue_in的末尾元素

        # 写法二:
        if self.empty():
            return None

        for i in range(len(self.queue_in) - 1):
            self.queue_out.append(self.queue_in.popleft())
        
        self.queue_in, self.queue_out = self.queue_out, self.queue_in 
        temp = self.queue_out.popleft()   
        self.queue_in.append(temp)
        return temp


    def empty(self) -> bool:
        """
        因为只有in存了数据,只要判断in是不是有数即可
        """
        return len(self.queue_in) == 0

方法三:用队列实现栈(此方法并没有模拟队列的特性,先进先出)

from collections import deque
import queue
class MyStack:
    def __init__(self) -> None:
        self.stack = deque()

    # 元素 x 入栈
    def push(self,x:int) -> None:
        self.stack.append(x)

    # 移除栈顶元素
    def pop(self) -> None:
        if self.empty():
            return None
        else:
            return self.stack.pop()

    # 获取栈顶元素

    def top(self) -> None:
        if self.empty():
            return None
        else:
            ans = self.stack.pop()
            self.stack.append(ans)
            return ans
    
    # 返回栈是否为空
    def empty(self) ->None:

        return not self.stack

 方法四:使用列表实现栈(此方法并没有模拟队列的特性,先进先出)

class MyStack:

    def __init__(self):
        self.stk = []

    def push(self, x: int) -> None:
        self.stk.append(x)

    def pop(self) -> int:
        tp = self.top()
        self.stk.pop()

        return tp

    def top(self) -> int:
        return self.stk[-1] 

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

作者:aJupyter
链接:https://leetcode.cn/problems/implement-stack-using-queues/solutions/2665064/225-yong-dui-lie-shi-xian-zhan-by-ajupyt-qhld/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

心得收获 

前两种方法是模拟队列的特性(先进先出),然后去实现栈,后两种方法是直接实现栈,更加简洁,但是本题目考查是用队列实现栈,希望大家能掌握了解相应的队列和栈的知识点,而并不是工程实践的意义。

参考文章

关于python中deque的基本语法可以参考下面的文章:

Python:【基础语法】 deque()用法_python deque-CSDN博客

  • 3
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是使用两个队列实现的 Kotlin 代码: ``` import java.util.* class MyStack() { private val queue1 = LinkedList<Int>() private val queue2 = LinkedList<Int>() private var topElement: Int? = null /** Push element x onto stack. */ fun push(x: Int) { queue2.offer(x) topElement = x while (!queue1.isEmpty()) { queue2.offer(queue1.poll()) } val temp = queue1 queue1 = queue2 queue2 = temp } /** Removes the element on top of the stack and returns that element. */ fun pop(): Int { val popped = queue1.poll() if (queue1.isEmpty()) { topElement = null } else { topElement = queue1.peek() } return popped } /** Get the top element. */ fun top(): Int { return topElement!! } /** Returns whether the stack is empty. */ fun empty(): Boolean { return queue1.isEmpty() } } ``` 解释一下代码: 我们使用两个队列 `queue1` 和 `queue2` 来模拟。`queue1` 存储中的元素,而 `queue2` 用来在 `push` 操作时暂存元素。 在 `push` 操作中,我们首先把元素 `x` 加入到 `queue2` 中,然后将 `queue1` 中的所有元素依次加入到 `queue2` 中。这样做的目的是为了让 `queue2` 中的元素按照的顺序排列,即最后加入的元素在队首,而最先加入的元素在队尾。然后我们将 `queue2` 和 `queue1` 交换,这样 `queue1` 中的元素就按照的顺序排列了。 在 `pop` 操作中,我们直接从 `queue1` 中弹出队首元素即可。如果 `queue1` 变为空队列,则将 `topElement` 置为 `null`。 在 `top` 操作中,我们返回 `topElement`,它记录了顶元素的值。 在 `empty` 操作中,我们判断 `queue1` 是否为空即可。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值