【小Y学算法】⚡️每日LeetCode打卡⚡️——41. 最小栈

简介: 📢前言🌲原题样例: 最小栈🌻C#方法一:迭代🌻Java方法:辅助栈💬总结

📢前言

🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻

🌲 每天打卡一道算法题,既是一个学习过程,又是一个分享的过程😜

🌲 提示:本专栏解题 编程语言一律使用 C# 和 Java 两种进行解题

🌲 要保持一个每天都在学习的状态,让我们一起努力成为算法大神吧🧐!

🌲 今天是力扣算法题持续打卡第41天🎈!

🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻🌻

🌲原题样例: 最小栈

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


push(x) —— 将元素 x 推入栈中。

pop() —— 删除栈顶的元素。

top()—— 获取栈顶元素。

getMin() —— 检索栈中的最小元素。

示例:

输入:
["MinStack","push","push","push","getMin","pop","top","getMin"]
[[],[-2],[0],[-3],[],[],[],[]]
输出:
[null,null,null,null,-3,null,0,-2]
解释:
MinStack minStack = new MinStack();
minStack.push(-2);
minStack.push(0);
minStack.push(-3);
minStack.getMin();   --> 返回 -3.
minStack.pop();
minStack.top();      --> 返回 0.
minStack.getMin();   --> 返回 -2.

提示:


pop、top 和 getMin 操作总是在 非空栈 上调用。


🌻C#方法一:迭代

仅使用一个栈完成后续遍历。步骤如下:


一直往左节点访问;

按根节点->右节点->左节点的顺序将节点依次压入栈中;

碰到叶子节点即开始出栈;

如果当前栈顶元素指向的节点依然有未访问到的子节点则回到步骤1,往复循环;

栈空,则遍历结束。

难点在于如何判断栈顶节点是否有未访问的子节点。

如果判断方式不当,很可能会因为栈顶节点是上一个已出栈节点的父节点,而导致其节点反复入栈出栈陷入死循环。


可以增加一个变量或指针,用于指向前一个已经出栈的节点。


每次判断栈顶节点的时候只需要提前判断一次是否与上一个出栈节点是父子关系,是的话则继续出栈,否则回到上面的步骤1再进行入栈循环。

public class MinStack {
    private Stack<int> _stack;
    private Stack<int> _minStack;
    /** initialize your data structure here. */
    public MinStack() {
        _stack = new Stack<int>();
        _minStack = new Stack<int>();
    }
    public void Push(int val) {
        _stack.Push(val);
        if(_minStack.Count > 0)
            _minStack.Push(Math.Min(val, _minStack.Peek()));
        else
            _minStack.Push(val);
    }
    public void Pop() {
        _minStack.Pop();
        _stack.Pop();
    }
    public int Top() {
        return _stack.Peek();
    }
    public int GetMin() {
        return _minStack.Peek();
    }
}

执行结果

执行通过

执行用时 132 ms, 在所有 Java 提交中击败了48.59%的用户
内存消耗 35.1 MB, 在所有 Java 提交中击败了15.82%的用户

🌻Java方法:辅助栈

思路和算法

要做出这道题目,首先要理解栈结构先进后出的性质。


对于栈来说,如果一个元素 a 在入栈时,栈里有其它的元素 b, c, d,那么无论这个栈在之后经历了什么操作,只要 a 在栈中,b, c, d 就一定在栈中,因为在 a 被弹出之前,b, c, d 不会被弹出。


因此,在操作过程中的任意一个时刻,只要栈顶的元素是 a,那么我们就可以确定栈里面现在的元素一定是 a, b, c, d。


那么,我们可以在每个元素 a 入栈时把当前栈的最小值 m 存储起来。在这之后无论何时,如果栈顶元素是 a,我们就可以直接返回存储的最小值 m。


image.gif

网络异常,图片无法展示
|

按照上面的思路,我们只需要设计一个数据结构,使得每个元素 a 与其相应的最小值 m 时刻保持一一对应。


因此我们可以使用一个辅助栈,与元素栈同步插入与删除,用于存储与每个元素对应的最小值。


当一个元素要入栈时,我们取当前辅助栈的栈顶存储的最小值,与当前元素比较得出最小值,将这个最小值插入辅助栈中;

当一个元素要出栈时,我们把辅助栈的栈顶元素也一并弹出;

在任意一个时刻,栈内元素的最小值就存储在辅助栈的栈顶元素中。

class MinStack {
    Deque<Integer> xStack;
    Deque<Integer> minStack;
    public MinStack() {
        xStack = new LinkedList<Integer>();
        minStack = new LinkedList<Integer>();
        minStack.push(Integer.MAX_VALUE);
    }
    public void push(int x) {
        xStack.push(x);
        minStack.push(Math.min(minStack.peek(), x));
    }
    public void pop() {
        xStack.pop();
        minStack.pop();
    }
    public int top() {
        return xStack.peek();
    }
    public int getMin() {
        return minStack.peek();
    }
}

执行结果

执行通过

执行用时 5 ms, 在所有 Java 提交中击败了78.78%的用户
内存消耗 39.8 MB, 在所有 Java 提交中击败了92.25%的用户

💬总结

  • 今天是力扣算法题打卡的第四十一天!
  • 文章采用 C# 和 Java 两种编程语言进行解题
  • 一些方法也是参考力扣大神写的,也是边学习边分享,再次感谢算法大佬们
  • 那今天的算法题分享到此结束啦,明天再见!
相关文章
|
18天前
|
算法
Leetcode 初级算法 --- 数组篇
Leetcode 初级算法 --- 数组篇
36 0
|
8天前
|
缓存 算法 Java
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS
这篇文章详细介绍了Java虚拟机(JVM)中的垃圾回收机制,包括垃圾的定义、垃圾回收算法、堆内存的逻辑分区、对象的内存分配和回收过程,以及不同垃圾回收器的工作原理和参数设置。
29 4
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS
|
11天前
|
算法 程序员 索引
数据结构与算法学习七:栈、数组模拟栈、单链表模拟栈、栈应用实例 实现 综合计算器
栈的基本概念、应用场景以及如何使用数组和单链表模拟栈,并展示了如何利用栈和中缀表达式实现一个综合计算器。
16 1
数据结构与算法学习七:栈、数组模拟栈、单链表模拟栈、栈应用实例 实现 综合计算器
|
5天前
|
算法
每日一道算法题(Leetcode 20)
每日一道算法题(Leetcode 20)
15 2
|
11天前
|
算法
数据结构与算法二:栈、前缀、中缀、后缀表达式、中缀表达式转换为后缀表达式
这篇文章讲解了栈的基本概念及其应用,并详细介绍了中缀表达式转换为后缀表达式的算法和实现步骤。
28 3
|
12天前
【LeetCode 24】225.用队列实现栈
【LeetCode 24】225.用队列实现栈
8 0
|
13天前
|
算法
【LeetCode 23】232.用栈实现队列
【LeetCode 23】232.用栈实现队列
14 0
|
18天前
|
算法 C++
【算法单调栈】 矩形牛棚(C/C++)
【算法单调栈】 矩形牛棚(C/C++)
|
2月前
|
算法
测试工程师的技能升级:LeetCode算法挑战与职业成长
这篇文章通过作者亲身体验LeetCode算法题的过程,探讨了测试工程师学习算法的重要性,并强调了算法技能对于测试职业成长的必要性。
57 1
测试工程师的技能升级:LeetCode算法挑战与职业成长
|
2月前
|
存储 算法 Java
LeetCode经典算法题:打家劫舍java详解
LeetCode经典算法题:打家劫舍java详解
63 2