LeetCode #155 最小栈

本贴最后更新于 1591 天前,其中的信息可能已经东海扬尘

155.png

Problem Description

设计一个支持 push ,pop ,top 操作,并能在常数时间内检索到最小元素的栈。

  • push(x) —— 将元素 x 推入栈中。
  • pop() —— 删除栈顶的元素。
  • top() —— 获取栈顶元素。
  • getMin() —— 检索栈中的最小元素。

e.g.

  1. MinStack minStack = new MinStack();
  2. minStack.push(-2);
  3. minStack.push(0);
  4. minStack.push(-3);
  5. minStack.getMin(); ==> 返回 -3.
  6. minStack.pop();
  7. minStack.top(); ==> 返回 0.
  8. minStack.getMin(); ==> 返回 -2.

Solution

该题很简单,但是有意思的是,我的解法也算通过了,虽然题目说

在常数时间内检索到最小元素的栈。

但是这里应该指的是 getMin() 这个方法,所以即使我的 push(Object obj) 方法复杂度比较高也没关系。。

因为有删除元素(pop 方法)的存在,所以最小值在 push 完之后还是会随时变。所以单纯使用一个变量来保存最小值是不可能的。

我使用了双栈来实现,实际不用双栈用其他的数据结构也可以。大概原理就是在 push 的过程中,一个标准 stack 正常 push,另一个辅助栈从栈底到栈顶从小到大排序的插入,利用了 Stack 的父类(Vector)方法 insertElementAt(Object obj, int i),这里注意 index 是第二个参数。pop 时,标准 stack 正常 pop,辅助 stack 只需要 remove 标准 stack 刚刚 pop 返回的对象即可 minStack.remove(stack.pop());。这样确保两栈移除的元素是相同的。

代码如下:

public class MinStack {
  
    private Stack<Integer> stack;
    private Stack<Integer> minStack;
    /**
     * 执行用时 : 6 ms , 在所有 Java 提交中击败了 97.91% 的用户
     * 内存消耗 : 41.7 MB , 在所有 Java 提交中击败了 14.46% 的用户
     */
    public MinStack() {
        stack = new Stack<>();
        minStack = new Stack<>();
    }

    public void push(int x) {
        stack.push(x);
        if (minStack.empty()) {
            minStack.push(x);
        } else if (x >= minStack.peek()) {
            minStack.push(x);
        } else {
            for (int i = 0; i <= minStack.size(); i++) {
                if (x < minStack.get(i)) {
                    minStack.insertElementAt(x, i);
                    break;
                }
            }
        }

    }

    public void pop() {
        minStack.remove(stack.pop());
    }

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

    public int getMin() {
        return minStack.firstElement();
    }

    @Override
    public String toString() {
        return "MinStack{" +
                "stack=" + stack +
                ", minStack=" + minStack +
                '}';
    }

    public static void main(String[] args) {

        MinStack minStack = new MinStack();
        minStack.push(-2);
        minStack.push(0);
        minStack.push(-3);
        minStack.push(-11);
        minStack.push(-1);
        System.out.println(minStack.toString());
        System.out.println(minStack.getMin());
        minStack.pop();
        minStack.top();
        System.out.println(minStack.getMin());

    }

}

Feature

虽然我这样做是通过的,但其实我想的不够多,看了高赞的题解,即使是利用辅助栈 push 方法时间复杂度也可以不到线性级 O(n)。

因为题目是说了设计的该栈对于移除元素只有 pop 方法,就是从栈顶移除,也就是说 next 元素比当前顶部元素大的话,就没必要保留了,因为在只有 pop 方法的情况下,该元素一定是先被移除的。所以在条件下辅助栈可以不需要把所有元素保存。

除非题目加一个删除中间某个元素的方法的条件。

stack.png

当然还有不用辅助栈的做法,利用压栈。

  • Java

    Java 是一种可以撰写跨平台应用软件的面向对象的程序设计语言,是由 Sun Microsystems 公司于 1995 年 5 月推出的。Java 技术具有卓越的通用性、高效性、平台移植性和安全性。

    3190 引用 • 8214 回帖 • 1 关注
  • 算法
    428 引用 • 254 回帖 • 24 关注
  • 7 引用
  • LeetCode

    LeetCode(力扣)是一个全球极客挚爱的高质量技术成长平台,想要学习和提升专业能力从这里开始,充足技术干货等你来啃,轻松拿下 Dream Offer!

    209 引用 • 72 回帖

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...