news 2026/5/6 4:46:27

【每日算法】LeetCode 155. 最小栈

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
【每日算法】LeetCode 155. 最小栈

对前端开发者而言,学习算法绝非为了“炫技”。它是你从“页面构建者”迈向“复杂系统设计者”的关键阶梯。它将你的编码能力从“实现功能”提升到“设计优雅、高效解决方案”的层面。从现在开始,每天投入一小段时间,结合前端场景去理解和练习,你将会感受到自身技术视野和问题解决能力的质的飞跃。------ 算法:资深前端开发者的进阶引擎

LeetCode 155. 最小栈

1. 题目描述

1.1 原题链接与基本信息

  • 题目链接:LeetCode 155. 最小栈
  • 难度:简单
  • 标签:栈、设计

1.2 题目描述与示例

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

实现MinStack类:

  • MinStack()初始化堆栈对象。
  • void push(int val)将元素val推入堆栈。
  • void pop()删除堆栈顶部的元素。
  • int top()获取堆栈顶部的元素。
  • 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.

提示:

  • -2^31 <= val <= 2^31 - 1
  • poptopgetMin操作总是在非空栈上调用。
  • 最多调用3 * 10^4pushpoptopgetMin

2. 问题分析

2.1 问题核心

我们需要实现一个特殊的栈,它除了支持常规的栈操作(pushpoptop)外,还需要支持在常数时间内返回当前栈中的最小元素。

2.2 难点分析

常规的栈结构可以在常数时间内完成pushpoptop操作,但是要获取最小值,通常需要遍历整个栈,时间复杂度为 O(n),不符合题目要求。因此,我们需要在数据结构和算法上进行设计,使得getMin操作的时间复杂度为 O(1)。

3. 解题思路

3.1 思路一:辅助栈法(最优解)

使用两个栈:一个主栈用于存储所有元素,另一个辅助栈用于存储主栈每个状态下的最小值。当元素入栈时,主栈正常压入,辅助栈则压入当前主栈的最小值(即辅助栈栈顶和当前值中的较小者)。出栈时,两个栈同时弹出栈顶元素。这样,辅助栈的栈顶始终对应主栈当前状态的最小值。

3.2 思路二:单个栈存储差值法

只使用一个栈,存储每个元素与当前最小值的差值。同时用一个变量min记录当前最小值。入栈时,压入差值并更新最小值;出栈时,根据栈顶差值反向更新最小值。这种方法节省了空间,但实现稍复杂,且需要注意整数溢出问题(JavaScript 使用 Number 类型,安全整数范围为-2^53 ~ 2^53,题目值范围在 32 位整数内,因此安全)。

3.3 思路三:节点存储当前最小值法

定义一个节点结构,每个节点存储当前值以及到该节点为止的最小值。这样,栈中每个元素都携带了当前状态的最小值信息。出栈操作不影响其他节点的最小值。这种方法同样可以实现常数时间获取最小值,但每个节点需要额外存储一个最小值,空间复杂度为 O(n)。

4. 代码实现

4.1 辅助栈法(JavaScript实现)

classMinStack{constructor(){this.stack=[];this.minStack=[];// 辅助栈,与主栈同步操作,栈顶存储当前主栈中的最小值}push(val){this.stack.push(val);// 如果辅助栈为空,或者当前值小于等于辅助栈栈顶,则压入当前值;否则重复压入辅助栈栈顶if(this.minStack.length===0||val<=this.minStack[this.minStack.length-1]){this.minStack.push(val);}else{this.minStack.push(this.minStack[this.minStack.length-1]);}}pop(){if(this.stack.length===0)return;this.stack.pop();this.minStack.pop();}top(){returnthis.stack[this.stack.length-1];}getMin(){returnthis.minStack[this.minStack.length-1];}}

4.2 差值法(JavaScript实现)

classMinStack{constructor(){this.stack=[];this.min=Infinity;// 当前最小值}push(val){if(this.stack.length===0){this.stack.push(0);// 差值为0this.min=val;}else{constdiff=val-this.min;this.stack.push(diff);// 如果差值小于0,说明val比当前最小值还小,更新最小值if(diff<0){this.min=val;}}}pop(){if(this.stack.length===0)return;constdiff=this.stack.pop();// 如果差值小于0,说明当前栈顶元素就是最小值,出栈后需要恢复前一个最小值if(diff<0){this.min=this.min-diff;// 因为 diff = val - oldMin, 且 val 就是当前的 min,所以 oldMin = min - diff}// 如果栈为空,重置minif(this.stack.length===0){this.min=Infinity;}}top(){constdiff=this.stack[this.stack.length-1];// 如果差值大于等于0,说明当前栈顶元素不是最小值,真实值为 min + diff// 如果差值小于0,说明当前栈顶元素是最小值,真实值就是 minreturndiff>=0?this.min+diff:this.min;}getMin(){returnthis.min;}}

4.3 节点存储法(JavaScript实现)

classMinStack{constructor(){this.stack=[];// 栈中每个元素是一个对象,包含val和当前最小值}push(val){if(this.stack.length===0){this.stack.push({val,min:val});}else{constcurrentMin=this.stack[this.stack.length-1].min;this.stack.push({val,min:Math.min(currentMin,val)});}}pop(){this.stack.pop();}top(){returnthis.stack[this.stack.length-1].val;}getMin(){returnthis.stack[this.stack.length-1].min;}}

5. 复杂度与优缺点对比

5.1 复杂度分析

方法时间复杂度 (push/pop/top/getMin)空间复杂度
辅助栈法O(1)O(n)
差值法O(1)O(n) (最坏情况下与辅助栈法相同,但平均情况下节省空间)
节点存储法O(1)O(n)

5.2 优缺点对比表格

方法优点缺点
辅助栈法思路直观,易于理解和实现;操作同步,逻辑简单。需要额外的栈,最坏情况下空间加倍。
差值法只使用一个栈,节省了空间(尤其当元素较多且最小值变化不大时)。实现稍复杂,需要考虑整数溢出(在JavaScript中问题不大);存储差值可能带来精度问题(但题目为整数,所以安全)。
节点存储法每个节点独立存储最小值,逻辑清晰;不需要额外数据结构。每个节点需要额外存储一个最小值,空间开销与辅助栈法类似。

最优解建议:辅助栈法。因其实现简单,可读性强,在大多数情况下是首选。虽然空间复杂度为 O(n),但实际应用中通常可以接受。

6. 总结

最小栈问题是一个经典的栈结构设计问题,其核心在于如何在维护栈的先进后出特性的同时,快速获取当前状态下的最小值。通过辅助栈、差值法或节点存储法,我们都可以在 O(1) 时间复杂度内完成所有操作。其中,辅助栈法是最直接和常用的方法,适合在面试和实际开发中快速实现。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/19 1:32:19

76、量子点细胞自动机乘法器与除法器详解

量子点细胞自动机乘法器与除法器详解 在当今科技飞速发展的时代,量子计算领域的研究日益深入,量子点细胞自动机(QCA)作为其中的重要组成部分,其乘法器和除法器的设计与实现备受关注。下面将详细介绍QCA乘法器和除法器。 1. QCA乘法器 乘法器在信号处理等众多应用中有着…

作者头像 李华
网站建设 2026/4/30 18:07:10

情感语音合成难点破解——EmotiVoice给出标准答案

情感语音合成的破局之路&#xff1a;EmotiVoice 如何让机器“动情” 在虚拟偶像直播中突然哽咽落泪&#xff0c;在智能助手中听到亲人般温柔的语调&#xff0c;在游戏NPC口中感受到真实的愤怒与嘲讽——这些曾属于科幻电影的情节&#xff0c;正随着情感语音合成技术的突破悄然走…

作者头像 李华
网站建设 2026/5/4 12:50:47

SlopeCraft终极指南:快速创建Minecraft立体地图画的艺术

SlopeCraft终极指南&#xff1a;快速创建Minecraft立体地图画的艺术 【免费下载链接】SlopeCraft Map Pixel Art Generator for Minecraft 项目地址: https://gitcode.com/gh_mirrors/sl/SlopeCraft 想要在Minecraft中打造令人惊叹的立体地图画吗&#xff1f;SlopeCraft…

作者头像 李华
网站建设 2026/5/3 10:26:41

42、IPv6与Fedora Linux网络安装全解析

IPv6与Fedora Linux网络安装全解析 1. IPv6相关操作 1.1 IPv6链路本地地址使用 在使用链路本地地址时,必须像使用OpenSSH一样,用百分号指定本地接口。不过目前, scp 和OpenSSH的手册页都未对这种特殊的IPv6语法进行描述。 1.2 IPv6自动配置 若想实现IPv6的自动配置,…

作者头像 李华
网站建设 2026/5/3 12:45:14

WordPress处理站群平台word文档批量上传

要求&#xff1a;开源&#xff0c;免费&#xff0c;技术支持 博客&#xff1a;WordPress 开发语言&#xff1a;PHP 数据库&#xff1a;MySQL 功能&#xff1a;导入Word,导入Excel,导入PPT(PowerPoint),导入PDF,复制粘贴word,导入微信公众号内容,web截屏 平台&#xff1a;Window…

作者头像 李华