LeetCode155:最小棧,最簡單的中等難度題,時間擊敗100%,記憶體也低於官方

来源:https://www.cnblogs.com/bolingcavalry/archive/2023/09/11/17689777.html
-Advertisement-
Play Games

官方代碼是直接使用JDK的Deque對象,這樣的代碼能學到什麼?熟練操作API嗎?還是自己實現一個最小棧吧,用時擊敗100%,記憶體擊敗78% ...


歡迎訪問我的GitHub

這裡分類和彙總了欣宸的全部原創(含配套源碼):https://github.com/zq2599/blog_demos

本篇概覽

  • 最近運氣不錯,在LeetCode上白撿一道送分題,官方設定的難度是中等,然而此題難度放在簡單的題庫中都是墊底的存在,對於刷題數太少的欣宸而言,這簡直就是力扣的饋贈,建議大家也不要錯過,花上幾分鐘將其拿下
  • 不嘮嗑了,下麵咱們一起來刷之
  • 為了提起您的興趣,這裡提前劇透一下:
  1. 用最簡單的數據結構-數組,來存儲數據,代碼整體非常簡單,適合新手閱讀
  2. 執行用時執行用時3毫秒, 在所有 Java 提交中擊敗了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%
    在這裡插入圖片描述
  • 接下來該我了,自己實現棧

題目分析

  • 前面廢話太多,這裡精簡一下,說說我理解的此題的重點
  1. 數據結構:怎麼存數據,才能保證高效的讀寫速度?
  2. 最小值問題:本題不僅要有基本的棧功能,還要時刻能返回棧內的最小值
  3. 記憶體怎麼優化?
  4. 耗時怎麼優化?
  • 接下來針對上述問題,逐個考慮

問題一:數據結構設計

  • 最高效的存取,一般是數組和鏈表,在java中,原始類型的數組更簡單,而鏈表就要用到對象了,相對更複雜,所以,數組是首選,至於用數組實現後進先出的棧,那也簡單嘛,用個int型變數做指針即可
  • 如下圖,例如固定長度10的數組,裡面存了兩個有效數據,int型變數pointer等於1即可,表示數組的位置1存儲著最後一個有效數據,後面再加入新的數據時,放在pointer+1的位置即可
    在這裡插入圖片描述

最小值問題

  • 題目要求中規定了getMin方法要返回當前棧內的最小值,所以我們要搞清楚什麼時候最小值會發生變化:
  1. 棧內增加元素時,可能新增的元素比棧內元素都小
  2. 棧內彈出元素時,可能彈出的元素是最小的那個
  • 對於增加元素時的處理最簡單:準備個成員變數min,每次增加元素時,都比較增加的元素和當前min誰最小,最小的更新到min中
  • 但是,彈出時呢?最小值被彈出去了,那麼原本次小的就成了最小的,但是次小的咱們沒存啊,這時候有兩種選擇:
  1. 首先,參考官方源碼,再準備一個數組,每次增加時,就把最小值放進來
  2. 其次,每次彈出時,再重新算一遍最小值,O(n)的耗時,感覺還好...
  • 斟酌再三,方案一會導致記憶體翻倍,所以還是優先考慮方案二吧,也就是每次彈出時重新計算一遍最小值

記憶體怎麼優化?

  • 接下來要考慮如何少使用記憶體
  • 首先要搞清楚的是:準備多大的數組才能滿足題目要求,官方說明如下圖,註意紅色箭頭,如果調用三萬次push,那就說明會存三萬個int數字,所以數組長度如果低於三萬,提交後就可能報錯,達不到官方要求
    在這裡插入圖片描述
  • 所以,數組的長度固定是三萬嗎?
  • 當然不需要,提交代碼後,LeetCode會執行多個用例,不可能每個用例都push三萬次的,所以固定三萬的長度,看似保險,實則浪費
  • 所以,優化思路可以借鑒HashMap的源碼:存不下的時候,就擴容,也就是準備一個新數組,把老數組的數據複製進去,至於擴多少?邏輯不必太複雜,翻倍即可

耗時怎麼優化

  • 還有最後一個問題要考慮:時間還能優化嗎?
  • 要想優化時間,首先咱們要知道哪裡會耗時,回顧前面的設計,最耗時的地方應該是彈出元素的時候,這時候要重新計算最小值,時間複雜度是O(n),每次彈出都要執行,有沒有可能優化一下呢?
  • 考慮下圖這種情況,棧內數據是1、2、3,其中1在棧頂,那麼彈出1之後,自然要在2和3中尋找最小值作為棧的最小值了,這是必要操作,不能優化
    在這裡插入圖片描述
  • 但是下圖這種情況呢?3在棧頂,彈出去之後,1還是最小值,此時就沒有必要重新比較一遍了
    在這裡插入圖片描述
  • 好了,分析和設計都完成了,也該寫代碼驗證效果了,我有種預感,自己設計的棧,比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,成績如下,用時和記憶體雙雙優於官方,尤其是用時,擊敗百分百!
    在這裡插入圖片描述
  • 至此,第155題順利完成,自我感覺是有一些收穫的,至少比官方的面向API編程收穫更大,更何況成績比官方的還好一些...

歡迎關註博客園:程式員欣宸

學習路上,你不孤單,欣宸原創一路相伴...


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 對於有科班背景的讀者,可以跳過本系列文章。這些文章的主要目的是通過簡單易懂的彙總,幫助非科班出身的讀者理解底層知識,進一步瞭解為什麼在面試中會涉及這些底層問題。否則,某些概念將始終無法理解。這些電腦基礎文章將為你打通知識的任督二脈,祝你在編程領域中取得成功! ...
  • 下麵的系列文章記錄瞭如何使用一塊linux開發扳和一塊OLED屏幕實現視頻的播放: 項目介紹 為OLED屏幕開發I2C驅動 使用cuda編程加速視頻處理 這是此系列文章的第2篇, 主要總結和記錄一個I2C從設備的驅動, 在linux內核中如何實現, 如何給用戶態的程式暴露合適的介面, 讓用戶態有機會 ...
  • 1. 除非遇到異常情況,否則不需要調整配置 1.1. 不要“調優”伺服器,不要使用比率、公式或“調優腳本”作為設置配置變數的基礎 1.1.1. 在互聯網上搜索配置建議並不總是一個好主意,你會在博客、論壇等找到很多糟糕的建議 1.1.2. 很難判斷誰是真正的專家 1.1.3. 不要相信流行的記憶體消耗公 ...
  • Flink是一個分散式系統,要求有效地分配和管理計算資源以執行流式應用程式。它集成了所有常見的集群資源管理器,如Hadoop YARN和Kubernetes,但也可以設置為作為standalone甚至庫運行。 本節概述了Flink的體繫結構,並描述了其主要組件如何交互以執行應用程式以及從故障中恢復。 ...
  • 1. mysql資料庫四種常見資料庫引擎 1. MyISAM: MyISAM是MySQL最早的資料庫引擎之一。它被設計成處理大量的插入和查詢操作。MyISAM表格的數據存儲在三個文件上:.frm文件存儲表結構,.MYD文件存儲數據,.MYI文件存儲索引。MyISAM表格不支持事務處理和崩潰恢復,因此 ...
  • 隨著需求的不斷開發,前端項目不斷膨脹,業務提出:你們的首頁載入也太慢啦,我都需要7、8秒才能看到內容,於是乎主管就讓我聯合後端開啟優化專項,目標是3s內展示完全首頁的內容。 性能指標 開啟優化時,我們要清晰的知道現狀和目標,以及我們採用什麼樣的手段,通過檢測什麼指標來查看到優化的過程。 結果指標 根 ...
  • 功能介紹 登錄 首頁 修改密碼 提交申請 提交列表 數據可視化 審核列表 前端 components結構 搭建Vue項目 ​ Vue3快速上手: ​ https://cn.vuejs.org/guide/quick-start.html#creating-a-vue-application 頁面佈局 ...
  • 溫馨提示:本文以vue3+vite+ts舉例,vite配置和ts語法側重較少,比較適合有vuex或者vue基礎的小伙伴們兒查閱。 安裝pinia yarn yarn add pinia npm npm install pinia pnpm pnpm add pinia 1-開始 方式一:在main. ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...