算法训练营第十天|LeetCode232、LeetCode225

文章介绍了如何使用两个栈来实现先入先出(FIFO)的队列,以及如何用两个队列来模拟后入先出(LIFO)的栈。在栈实现队列的过程中,利用两个栈,一个用于push和peek,另一个用于pop。在队列实现栈时,通过在push时向队列添加元素,pop时将所有元素移到另一个队列并返回顶部元素,从而达到栈的效果。
摘要由CSDN通过智能技术生成

232. 用栈实现队列
        请你仅使用两个栈实现先入先出队列。队列应当支持一般队列支持的所有操作(push、pop、peek、empty):

        实现 MyQueue 类:

                void push(int x) 将元素 x 推到队列的末尾
                int pop() 从队列的开头移除并返回元素
                int peek() 返回队列开头的元素
                boolean empty() 如果队列为空,返回 true ;否则,返回 false
        说明:

                你 只能 使用标准的栈操作 —— 也就是只有 push to top, peek/pop from top, size, 和 is empty 操作是合法的。
                你所使用的语言也许不支持栈。你可以使用 list 或者 deque(双端队列)来模拟一个栈 

2:解题思路
        这道题是要用栈来实现队列,区别在于栈是先进后出,队列是先进先出,因此要实现先进入栈的元素,先出来,所以我们可以用两个栈来进行操作。一个用来进行push(入栈),一个用来进行pop(出栈)。

        第一步:先初始化两个栈,stack_in = [],stack_out = []。

        第二步:实现void push(int x)函数,将x推到队列的末尾,即栈的末尾,列表的append()可以将元素添加到列表的末尾,stack_in.append(x)。

        第三步:实现int pop() 从队列的开头移除并返回元素

                首先判断队列是否为空,为空直接返回None。

                判断stack_out是否为空,不为空,则将stack_out中最后一个元素弹出(因为stack_out是用来出栈的,此时stack_out中的出栈顺序与队列的出队列的顺序一直,需要将stack_out中所有的元素全部弹出,才能往stack_out中添加新的元素,不然出栈的顺序就与队列的不一样了)。

                stack_out为空,则说明现在没有需要出栈的元素,将stack_in中的元素从后往前依次加入到stack_out中(因为要保持先进先出,stack_in后面的元素的后进入的,因此要先加入stack_out中,这样才能保证stack_out在出栈的时候,先进入元素先出来),stack_in中的元素全部加入stack_out中后,就可以使用列表的pop()函数,将最后一个元素(及先进入队列的元素)弹出,并返回元素的值。

        第四步:实现int peek() 返回队列开头的元素

                我们可以直接调用第三步实现的pop()函数,然后用一个变量承接弹出的元素(即队列开头的元素),再把弹出的元素,加回到stack_out中。

        第五步:boolean empty() 如果队列为空,返回 true ;否则,返回 false

                直接判断stack_in,stack_out其中一个栈是否为空,只要有一个不为空,则表明队列不为空,反之,队列为空。
 

class MyQueue:
 
    def __init__(self):
        self.stack_in = []        # 主要进行push-入栈
        self.stack_out = []       # 主要进行pop-出栈
 
 
    def push(self, x):
        self.stack_in.append(x)       # 往stack_in中push数字
 
 
    def pop(self):
        if self.empty():
            return None
 
        if self.stack_out:            # 当stack_out(出栈)不为空时
            return self.stack_out.pop()        # 将stack_out中的最后一个元素弹出(pop)
        else:                          # 当stack_out(出栈)为空时
            for i in range(len(self.stack_in)):         # 将stack_in中的元素全部添加到stack_out
                self.stack_out.append(self.stack_in.pop())      # 依次将stack_in中的最后一个元素弹出(pop),添加到stack_out中
            return self.stack_out.pop()                 # 将stack_out中最后一个元素弹出并返回,即是将队列的开头元素弹出并返回
 
 
    def peek(self):
        res = self.pop()              # 调用self.pop(),将stack_out中的最后一个元素弹出,即是将队列的开头元素弹出
        self.stack_out.append(res)    # 再把弹出的元素重新放到stack_out的最后位置
        return res                    # 返回弹出的最后一个元素
 
 
    def empty(self):
        if self.stack_out or self.stack_in:         # 当stack_out和stack_in中有一个不为空,返回False
            return False
        else:
            return True

225. 用队列实现栈

 请你仅使用两个队列实现一个后入先出(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(双端队列)来模拟一个队列 , 只要是标准的队列操作即可。

2:解题思路
        用队列来实现栈,就是要实现,先进入的元素后出,后进入的元素先出。

        解法一:使用两个队列来实现栈

        第一步:使用deque()初始化两个队列,queue_in(负责存数据),queue_out(在pop的时候用)

        第二步:实现void push(int x) 将元素 x 压入栈顶

                使用append(x),往queue_in中添加元素

        第三步:实现int pop() 移除并返回栈顶元素

                首先判断栈是否为空,为空则直接返回None

                然后获取存数据的队列queue_in的长度,我们需要将queue_in最后一个元素之前的元素按照从前往后依次加入到queue_out中(因为栈是后进入的元素,先出,所以需要将最后一个元素之前的元素依次加到另外一个队列中),这样queue_in中的元素就是需要弹出的元素(栈中最先弹出的元素)

                但是我们规定了queue_in是用来存数据的,queue_out是在pop()时使用的,所以我们将queue_in和queue_out两个队列进行交换,这样需要弹出的元素就在queue_out中了,我们再弹出这个元素。

        第四步:实现int top() 返回栈顶元素

                首先也需要判断栈是否为空,为空则直接返回None

                因为我们是用列表实现的队列,栈顶的元素就是列表的最后一个元素,我们直接返回列表的最后一个元素

        第五步:实现boolean empty() 如果栈是空的,返回 true ;否则,返回 false 

                因为一直是queue_in在存数据,所以我们只要判断queue_in的长度是否等于0,等于0则表示栈是空的,反之则不是空的。
 

class MyStack:
 
    def __init__(self):
        self.queue_in = deque()             # 初始化一个队列,往队列中存数据
        self.queue_out = deque()            # 在pop的时候使用
 
 
    def push(self, x):
        self.queue_in.append(x)        # 将元素入队列
 
 
    def pop(self):
        if self.empty():
            return None
        size = len(self.queue_in)           # 获取队列的长度
        # 将队列中最后一个元素之前的元素依次从队列中取出,再加入queue_out队列
        for i in range(size-1):
            self.queue_out.append(self.queue_in.popleft())
        
        # 将queue_in和queue_out进行交换,保证queue_in一直是存数据的
        self.queue_in, self.queue_out = self.queue_out, self.queue_in
        # 这样queue_out中剩余的最后一个元素就是栈顶的元素 
        return self.queue_out.popleft()
    def top(self):
        if self.empty():
            return None
        # 直接返回队列中的最后一个元素
        return self.queue_in[-1]
 
 
    def empty(self):
        if len(self.queue_in) == 0:        # 只要stack_in中存了数据,所以判断stack_in的长度为不为0即可
            return True
        else:
            return False

 解法二:使用一个队列来实现栈

        我们只需要将队列中最后一个元素之前的元素弹出,再将弹出的元素依次添加进队列,这样将最后一个元素之前的元素全部进行移动后,最后一个元素成为了第一个元素,此时弹出元素就实现了栈的后进入的元素,先出

        第一步:使用deque()初始化一个队列,queue

        第二步:实现void push(int x) 将元素 x 压入栈顶

                使用append(x),往queue中添加元素

        第三步:实现int pop() 移除并返回栈顶元素

                首先判断栈是否为空,为空则直接返回None

                然后获取存数据的队列queue的长度,我们需要将queue最后一个元素之前的元素按照从前往后依次弹出,再加入到queue中(因为栈是后进入的元素,先出,所以需要将最后一个元素移动到队列的开头位置),这样queue中的第一个元素就是需要弹出的元素(栈中最先弹出的元素),我们再弹出这个元素。

        第四步:实现int top() 返回栈顶元素

                首先也需要判断栈是否为空,为空则直接返回None

                因为我们是用列表实现的队列,栈顶的元素就是列表的最后一个元素,我们直接返回列表的最后一个元素

        第五步:实现boolean empty() 如果栈是空的,返回 true ;否则,返回 false 

   

class MyStack:
 
    def __init__(self):
        self.queue = deque()             # 初始化一个队列
 
    def push(self, x):
        self.queue.append(x)        # 将元素入队列
 
    def pop(self):
        if self.empty():
            return None
        size = len(self.queue)           # 获取队列的长度
        # 将队列中最后一个元素之前的元素依次从队列中取出,再加入队列
        for i in range(size-1):
            self.queue.append(self.queue.popleft())
        # 这样队列的第一个元素,就是栈顶元素
        return self.queue.popleft()
           
    def top(self):
        if self.empty():
            return None
        # 直接返回队列中的最后一个元素
        return self.queue[-1]
 
    def empty(self):
        if self.queue:
            return False
        else:
            return True

            只要判断queue是否为空,为空则表示栈是空的,反之则不是空的。

第二十二天的算法训练营主要涵盖了Leetcode题目中的三道题目,分别是Leetcode 28 "Find the Index of the First Occurrence in a String",Leetcode 977 "有序数组的平方",和Leetcode 209 "长度最小的子数组"。 首先是Leetcode 28题,题目要求在给定的字符串中找到第一个出现的字符的索引。思路是使用双指针来遍历字符串,一个指向字符串的开头,另一个指向字符串的结尾。通过比较两个指针所指向的字符是否相等来判断是否找到了第一个出现的字符。具体实现的代码如下: ```python def findIndex(self, s: str) -> int: left = 0 right = len(s) - 1 while left <= right: if s[left == s[right]: return left left += 1 right -= 1 return -1 ``` 接下来是Leetcode 977题,题目要求对给定的有序数组中的元素进行平方,并按照非递减的顺序返回结果。这里由于数组已经是有序的,所以可以使用双指针的方法来解决问题。一个指针指向数组的开头,另一个指针指向数组的末尾。通过比较两个指针所指向的元素的绝对值的大小来确定哪个元素的平方应该放在结果数组的末尾。具体实现的代码如下: ```python def sortedSquares(self, nums: List[int]) -> List[int]: left = 0 right = len(nums) - 1 ans = [] while left <= right: if abs(nums[left]) >= abs(nums[right]): ans.append(nums[left ** 2) left += 1 else: ans.append(nums[right ** 2) right -= 1 return ans[::-1] ``` 最后是Leetcode 209题,题目要求在给定的数组中找到长度最小的子数组,使得子数组的和大于等于给定的目标值。这里可以使用滑动窗口的方法来解决问题。使用两个指针来表示滑动窗口的左边界和右边界,通过移动指针来调整滑动窗口的大小,使得滑动窗口中的元素的和满足题目要求。具体实现的代码如下: ```python def minSubArrayLen(self, target: int, nums: List[int]) -> int: left = 0 right = 0 ans = float('inf') total = 0 while right < len(nums): total += nums[right] while total >= target: ans = min(ans, right - left + 1) total -= nums[left] left += 1 right += 1 return ans if ans != float('inf') else 0 ``` 以上就是第二十二天的算法训练营的内容。通过这些题目的练习,可以提升对双指针和滑动窗口等算法的理解和应用能力。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值