155. 最小栈
题目:设计一个支持 push ,pop ,top 操作,并能在常数时间内检索到最小元素的栈。
实现 MinStack 类:
MinStack() 初始化堆栈对象。
void push(int val) 将元素val推入堆栈。
void pop() 删除堆栈顶部的元素。
int top() 获取堆栈顶部的元素。
int getMin() 获取堆栈中的最小元素。
链接 https://leetcode.cn/problems/min-stack/
个人思路
- 设计一个栈min_stack来储存对应位置前面的最小值,当原来的栈stack添加新的值时,与min_stack栈顶元素比较谁小,min_stack。append(min(val,min_stack[-1])。当stack执行pop操作时,min_stack也弹出一个最小值。
class MinStack:
def __init__(self):
self.stack = []
self.min_stack = [math.inf]
def push(self, val: int) -> None:
self.stack.append(val)
if val < self.min_stack[-1]:
self.min_stack.append(val)
else:
self.min_stack.append(self.min_stack[-1])
def pop(self) -> None:
self.stack.pop()
self.min_stack.pop()
def top(self) -> int:
return self.stack[-1]
def getMin(self) -> int:
return self.min_stack[-1]
# Your MinStack object will be instantiated and called as such:
# obj = MinStack()
# obj.push(val)
# obj.pop()
# param_3 = obj.top()
# param_4 = obj.getMin()
时间复杂度:对于题目中的所有操作,时间复杂度均为 O(1)。因为栈的插入、删除与读取操作都是 O(1),我们定义的每个操作最多调用栈操作两次。
空间复杂度:O(n),其中 n 为总操作数。最坏情况下,我们会连续插入 n 个元素,此时两个栈占用的空间为 O(n)。
其他思路
评论区大佬:不用额外空间的做法。思路是栈里保存差值
-
- stack用来存储和min的差值,min存储最小值,每次出栈的时候通过差值和当前min计算要出栈的值和之前的min
- 如果差值diff大于等于0,说明要出栈的值大于等于当前min,那么要出栈的值在入栈的时候没有更新min,返回min+diff;
- 如果插值diff小于0,说明当前要出栈的值就是min(因为入栈的时候我们选择的就是min和入栈元素的最小值),同时,通过min-diff计算出之前min
- 要注意的是diff可能会超出int范围,类似于 Integer.MAX_VALUE - 1 这种,所以diff要用Long存(java)
class MinStack:
def __init__(self):
"""
initialize your data structure here.
"""
self.stack = []
self.min_value = -1
def push(self, x: int) -> None:
if not self.stack:
self.stack.append(0)
self.min_value = x
else:
diff = x-self.min_value
self.stack.append(diff)
self.min_value = self.min_value if diff > 0 else x
def pop(self) -> None:
if self.stack:
diff = self.stack.pop()
if diff < 0:
top = self.min_value
self.min_value = top - diff
else:
top = self.min_value + diff
return top
def top(self) -> int:
return self.min_value if self.stack[-1] < 0 else self.stack[-1] + self.min_value
def getMin(self) -> int:
return self.min_value if self.stack else -1
注意
突然想到,我是不是可以同时把当前需要插入的值和最小值都压入栈中,即stack[-1]为栈最小值,stack[-2]才是栈最顶端的值,尝试实现了一下,发现能通过测试:
class MinStack:
def __init__(self):
self.stack = []
def push(self, val: int) -> None:
self.stack.append(val)
# 将val和min分别压入栈中
if len(self.stack) < 2:
self.stack.append(val)
elif val < self.stack[-2]:
self.stack.append(val)
else:
self.stack.append(self.stack[-2])
def pop(self) -> None:
self.stack.pop()
self.stack.pop()
def top(self) -> int:
return self.stack[-2]
def getMin(self) -> int:
return self.stack[-1]
# Your MinStack object will be instantiated and called as such:
# obj = MinStack()
# obj.push(val)
# obj.pop()
# param_3 = obj.top()
# param_4 = obj.getMin()