LeetCode155:最小栈,最简单的中等难度题,时间击败100%,内存也低于官方

这篇具有很好参考价值的文章主要介绍了LeetCode155:最小栈,最简单的中等难度题,时间击败100%,内存也低于官方。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

欢迎访问我的GitHub

这里分类和汇总了欣宸的全部原创(含配套源码):https://github.com/zq2599/blog_demos

本篇概览

  • 最近运气不错,在LeetCode上白捡一道送分题,官方设定的难度是中等,然而此题难度放在简单的题库中都是垫底的存在,对于刷题数太少的欣宸而言,这简直就是力扣的馈赠,建议大家也不要错过,花上几分钟将其拿下
  • 不唠嗑了,下面咱们一起来刷之
  • 为了提起您的兴趣,这里提前剧透一下:
  1. 用最简单的数据结构-数组,来存储数据,代码整体非常简单,适合新手阅读
  2. 执行用时执行用时3毫秒, 在所有 Java 提交中击败了100%的用户(包括官方),有下图为证
    LeetCode155:最小栈,最简单的中等难度题,时间击败100%,内存也低于官方

题目说明

  • 设计一个支持 push ,pop ,top 操作,并能在常数时间内检索到最小元素的栈。
  • 实现 MinStack 类:
  1. MinStack() 初始化堆栈对象。
  2. void push(int val) 将元素val推入堆栈。
  3. void pop() 删除堆栈顶部的元素。
  4. int top() 获取堆栈顶部的元素。
  5. int getMin() 获取堆栈中的最小元素。
  • 示例1
输入:
["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.
  • 提示
  1. -231 <= val <= 231 - 1
  2. pop、top 和 getMin 操作总是在 非空栈 上调用
  3. push, pop, top, and getMin最多被调用 30000 次

官方解法

  • 先来看官方解法,简单的说,就是用JDK已有的栈对象Deque来完成push、pop、top等操作,如下所示
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();
    }
}
  • 读完这段代码我就茫然了:我来LeetCode刷题是为了什么?为了学习Deque类的API使用方法吗?
  • 不,我是来学习和提升自己的算法能力的,这种API调用并不是我心目中的答案,官方找不到,我就自己动手
  • 毕竟,实现个栈能有多大难度?
  • 为了弄清楚自己和官方的差距,我先将上述官方源码提交一次,看看成绩如何,如下图,官方在使用Deque类的情况下,执行用时击败93%,内存消耗击败57%
    LeetCode155:最小栈,最简单的中等难度题,时间击败100%,内存也低于官方
  • 接下来该我了,自己实现栈

题目分析

  • 前面废话太多,这里精简一下,说说我理解的此题的重点
  1. 数据结构:怎么存数据,才能保证高效的读写速度?
  2. 最小值问题:本题不仅要有基本的栈功能,还要时刻能返回栈内的最小值
  3. 内存怎么优化?
  4. 耗时怎么优化?
  • 接下来针对上述问题,逐个考虑

问题一:数据结构设计

  • 最高效的存取,一般是数组和链表,在java中,原始类型的数组更简单,而链表就要用到对象了,相对更复杂,所以,数组是首选,至于用数组实现后进先出的栈,那也简单嘛,用个int型变量做指针即可
  • 如下图,例如固定长度10的数组,里面存了两个有效数据,int型变量pointer等于1即可,表示数组的位置1存储着最后一个有效数据,后面再加入新的数据时,放在pointer+1的位置即可
    LeetCode155:最小栈,最简单的中等难度题,时间击败100%,内存也低于官方

最小值问题

  • 题目要求中规定了getMin方法要返回当前栈内的最小值,所以我们要搞清楚什么时候最小值会发生变化:
  1. 栈内增加元素时,可能新增的元素比栈内元素都小
  2. 栈内弹出元素时,可能弹出的元素是最小的那个
  • 对于增加元素时的处理最简单:准备个成员变量min,每次增加元素时,都比较增加的元素和当前min谁最小,最小的更新到min中
  • 但是,弹出时呢?最小值被弹出去了,那么原本次小的就成了最小的,但是次小的咱们没存啊,这时候有两种选择:
  1. 首先,参考官方源码,再准备一个数组,每次增加时,就把最小值放进来
  2. 其次,每次弹出时,再重新算一遍最小值,O(n)的耗时,感觉还好...
  • 斟酌再三,方案一会导致内存翻倍,所以还是优先考虑方案二吧,也就是每次弹出时重新计算一遍最小值

内存怎么优化?

  • 接下来要考虑如何少使用内存
  • 首先要搞清楚的是:准备多大的数组才能满足题目要求,官方说明如下图,注意红色箭头,如果调用三万次push,那就说明会存三万个int数字,所以数组长度如果低于三万,提交后就可能报错,达不到官方要求
    LeetCode155:最小栈,最简单的中等难度题,时间击败100%,内存也低于官方
  • 所以,数组的长度固定是三万吗?
  • 当然不需要,提交代码后,LeetCode会执行多个用例,不可能每个用例都push三万次的,所以固定三万的长度,看似保险,实则浪费
  • 所以,优化思路可以借鉴HashMap的源码:存不下的时候,就扩容,也就是准备一个新数组,把老数组的数据复制进去,至于扩多少?逻辑不必太复杂,翻倍即可

耗时怎么优化

  • 还有最后一个问题要考虑:时间还能优化吗?
  • 要想优化时间,首先咱们要知道哪里会耗时,回顾前面的设计,最耗时的地方应该是弹出元素的时候,这时候要重新计算最小值,时间复杂度是O(n),每次弹出都要执行,有没有可能优化一下呢?
  • 考虑下图这种情况,栈内数据是1、2、3,其中1在栈顶,那么弹出1之后,自然要在2和3中寻找最小值作为栈的最小值了,这是必要操作,不能优化
    LeetCode155:最小栈,最简单的中等难度题,时间击败100%,内存也低于官方
  • 但是下图这种情况呢?3在栈顶,弹出去之后,1还是最小值,此时就没有必要重新比较一遍了
    LeetCode155:最小栈,最简单的中等难度题,时间击败100%,内存也低于官方
  • 好了,分析和设计都完成了,也该写代码验证效果了,我有种预感,自己设计的栈,比LeetCode官方的更快,也更省内存,希望不要被现实打脸...

编码

  • 完整代码如下,有了前面的详细分析,相信您可以轻松看懂,注意我这里数组的初始长度是64,您也可以调整成其他值试试
class MinStack {

   private int[] array = new int[64];
    private int pointer = -1;

    private int min = Integer.MAX_VALUE;

    public MinStack() {

    }

    public void push(int val) {
        array[++pointer] = val;
        min = Math.min(min, val);

        // 扩容
        if (pointer==(array.length-1)) {
            int[] temp = new int[array.length*2];
            System.arraycopy(array, 0, temp, 0, array.length);
            array = temp;
        }
    }

    public void pop() {
        pointer--;

        // 这里可以优化:如果弹出的不是最小值,那就没必要重算呀!
        if (array[pointer+1]==min) {
            min = Integer.MAX_VALUE;
            for (int i=0;i<=pointer;i++) {
                min = Math.min(min, array[i]);
            }
        }
    }

    public int top() {
        return array[pointer];
    }

    public int getMin() {
        return min;
    }
}
  • 提交,顺利AC,成绩如下,用时和内存双双优于官方,尤其是用时,击败百分百!
    LeetCode155:最小栈,最简单的中等难度题,时间击败100%,内存也低于官方
  • 至此,第155题顺利完成,自我感觉是有一些收获的,至少比官方的面向API编程收获更大,更何况成绩比官方的还好一些...

欢迎关注博客园:程序员欣宸

学习路上,你不孤单,欣宸原创一路相伴...文章来源地址https://www.toymoban.com/news/detail-705530.html

到了这里,关于LeetCode155:最小栈,最简单的中等难度题,时间击败100%,内存也低于官方的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【Leetcode】155. 最小栈、JZ31 栈的压入、弹出序列

     作者:小卢   专栏:《Leetcode》 喜欢的话:世间因为少年的挺身而出,而更加瑰丽。                                  ——《人民日报》 155. 最小栈 题目描述; 设计一个支持 push ,pop ,top 操作,并能在常数时间内检索到最小元素的栈。 实现 MinStack 类: MinStack() 初始化

    2024年02月13日
    浏览(49)
  • LeetCode297:hard级别中最简单的存在,java版,用时击败98%,内存击败百分之九十九

    因为欣宸个人水平有限,在刷题时一直不敢面对hard级别的题目,生怕出现 一杯茶一包烟,一道hard做一天 的窘境 这种恐惧心理一直在,直到遇见了它:LeetCode297,建议不敢做hard题的新手们速来围观,拿它练手,轻松找到自信 二叉树的序列化与反序列化 提示 接下来,先开始

    2024年02月09日
    浏览(29)
  • LeetCode、875. 爱吃香蕉的珂珂【中等,最小速度二分】

    博主介绍:✌目前全网粉丝2W+,csdn博客专家、Java领域优质创作者,博客之星、阿里云平台优质作者、专注于Java后端技术领域。 涵盖技术内容:Java后端、算法、分布式微服务、中间件、前端、运维、ROS等。 博主所有博客文件目录索引:博客目录索引(持续更新) 视频平台:

    2024年01月24日
    浏览(63)
  • 每天一道leetcode:433. 最小基因变化(图论&中等&广度优先遍历)

    基因序列可以表示为一条由 8 个字符组成的字符串,其中每个字符都是 \\\'A\\\' 、 \\\'C\\\' 、 \\\'G\\\' 和 \\\'T\\\' 之一。 假设我们需要调查从基因序列 start 变为 end 所发生的基因变化。一次基因变化就意味着这个基因序列中的一个字符发生了变化。 例如, \\\"AACCGGTT\\\" -- \\\"AACCGGTA\\\" 就是一次基因变化。

    2024年02月12日
    浏览(41)
  • LeetCode、2462. 雇佣 K 位工人的总代价【中等,最小堆+双指针】

    博主介绍:✌目前全网粉丝2W+,csdn博客专家、Java领域优质创作者,博客之星、阿里云平台优质作者、专注于Java后端技术领域。 涵盖技术内容:Java后端、算法、分布式微服务、中间件、前端、运维、ROS等。 博主所有博客文件目录索引:博客目录索引(持续更新) 视频平台:

    2024年01月19日
    浏览(42)
  • LeetCode_二分搜索_中等_2594.修车的最少时间

    给你一个整数数组 ranks ,表示一些机械工的 能力值 。ranksi 是第 i 位机械工的能力值。能力值为 r 的机械工可以在 r * n 2 分钟内修好 n 辆车。 同时给你一个整数 cars ,表示总共需要修理的汽车数目。请你返回修理所有汽车最少需要多少时间。 注意:所有机械工可以同时修理

    2024年02月09日
    浏览(43)
  • windows内存取证-中等难度-下篇

    上文我们对第一台Target机器进行内存取证,今天我们继续往下学习,内存镜像请从上篇获取,这里不再进行赘述​ 攻击者执行了net use z: 10.1.1.2c$ 指令将 10.1.1.2域控制器的C盘映射到本地的Z盘,并且使用了rar压缩工具将文件存储在 crownjewlez.rar里,所以密码就在这里了 将进程

    2024年02月05日
    浏览(40)
  • 生成窗口最大值数组【中等难度】

    给定一个整数数组 nums 和一个正整数 k,滑动一个大小为 k 的窗口,从数组的左边到右边,找到每个窗口中的最大值。 示例 1: 输入:nums = {1, 3, -1, -3, 5, 3, 6, 7} 、k = 3 输出:3 3 5 5 6 7 我们可以使用双端队列(deque)来解决这个问题。双端队列可以在两端进行插入和删除操作,

    2024年02月12日
    浏览(48)
  • LeetCode_BFS_DFS_中等_1376.通知所有员工所需的时间

    公司里有 n 名员工,每个员工的 ID 都是独一无二的,编号从 0 到 n - 1。公司的总负责人通过 headID 进行标识。 在 manager 数组中,每个员工都有一个直属负责人,其中 manager[i] 是第 i 名员工的直属负责人。对于总负责人,manager[headID] = -1。题目保证从属关系可以用树结构显示。

    2024年02月02日
    浏览(28)
  • 趣说数据结构(练习2) —— 顺序表/链表力扣刷题(中等难度)

    力扣原题:https://leetcode.cn/problems/reverse-linked-list-ii/ 题目描述 给你单链表的头指针 head 和两个整数 left 和 right ,其中 left = right 。请你反转从位置 left 到位置 right 的链表节点,返回 反转后的链表 。 示例 1 输入:head = [1,2,3,4,5], left = 2, right = 4 输出:[1,4,3,2,5] 示例 2 输入:h

    2024年02月01日
    浏览(55)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包