代码随想录 day 48 单调栈

2024-08-21 02:36
文章标签 随想录 代码 day 单调 48

本文主要是介绍代码随想录 day 48 单调栈,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

第十章 单调栈part01

739. 每日温度

今天正式开始单调栈,这是单调栈一篇扫盲题目,也是经典题。
大家可以读题,思考暴力的解法,然后在看单调栈的解法。 就能感受出单调栈的巧妙
https://programmercarl.com/0739.%E6%AF%8F%E6%97%A5%E6%B8%A9%E5%BA%A6.html

496.下一个更大元素 I

本题和 739. 每日温度 看似差不多,其实 有加了点难度。
https://programmercarl.com/0496.%E4%B8%8B%E4%B8%80%E4%B8%AA%E6%9B%B4%E5%A4%A7%E5%85%83%E7%B4%A0I.html

503.下一个更大元素II

这道题和 739. 每日温度 几乎如出一辙,可以自己尝试做一做
https://programmercarl.com/0503.%E4%B8%8B%E4%B8%80%E4%B8%AA%E6%9B%B4%E5%A4%A7%E5%85%83%E7%B4%A0II.html

单调栈理论

我怎么能想到用单调栈呢? 什么时候用单调栈呢?
通常是一维数组,要寻找任一个元素的右边或者左边第一个比自己大或者小的元素的位置,此时我们就要想到可以用单调栈了。时间复杂度为O(n)。
例如本题其实就是找找到一个元素右边第一个比自己大的元素,此时就应该想到用单调栈了。
那么单调栈的原理是什么呢?为什么时间复杂度是O(n)就可以找到每一个元素的右边第一个比它大的元素位置呢?
单调栈的本质是空间换时间,因为在遍历的过程中需要用一个栈来记录右边第一个比当前元素高的元素,优点是整个数组只需要遍历一次。
更直白来说,就是用一个栈来记录我们遍历过的元素,因为我们遍历数组的时候,我们不知道之前都遍历了哪些元素,以至于遍历一个元素找不到是不是之前遍历过一个更小的,所以我们需要用一个容器(这里用单调栈)来记录我们遍历过的元素。
在使用单调栈的时候首先要明确如下几点:

  1. 单调栈里存放的元素是什么?

单调栈里只需要存放元素的下标i就可以了,如果需要使用对应的元素,直接T[i]就可以获取。

  1. 单调栈里元素是递增呢? 还是递减呢?

注意以下讲解中,顺序的描述为 从栈头到栈底的顺序,因为单纯的说从左到右或者从前到后,不说栈头朝哪个方向的话,大家一定比较懵。
这里我们要使用递增循序(再强调一下是指从栈头到栈底的顺序),因为只有递增的时候,栈里要加入一个元素i的时候,才知道栈顶元素在数组中右面第一个比栈顶元素大的元素是i。
即:如果求一个元素右边第一个更大元素,单调栈就是递增的,如果求一个元素右边第一个更小元素,单调栈就是递减的。
使用单调栈主要有三个判断条件。

  • 当前遍历的元素T[i]小于栈顶元素T[st.top()]的情况
  • 当前遍历的元素T[i]等于栈顶元素T[st.top()]的情况
  • 当前遍历的元素T[i]大于栈顶元素T[st.top()]的情况

把这三种情况分析清楚了,也就理解透彻了

739. 每日温度

题目链接

https://leetcode.cn/problems/daily-temperatures/description/

解题思路

本题求的是数组右边第一个比自己大的元素,所以根据单调栈理论,就是从栈头到栈底单调递增
自己想一想,栈就是记录遍历过的元素,一直没有找遍历元素大于栈顶的元素就一直加入,直到找到了,开始弹出栈顶元素,计算下标值 i-stack.pop() 就是据里,弹出一个后还要进行比较是否大于栈顶,此时用while循环重复上面逻辑。

code

class Solution {public int[] dailyTemperatures(int[] temperatures) {int[] res=new int[temperatures.length];//定义一个栈,存放的元素保证从栈顶到栈底单调递增Stack<Integer> stack=new Stack<>();//初始化栈第一个元素,栈里存放的是元素下标stack.push(0);//开始遍历元素for(int i=1;i<temperatures.length;i++){//12.当前遍历元素小于等于 加入栈 符合栈顶到栈底单调递增if(temperatures[i]<=temperatures[stack.peek()]){stack.push(i);}else{//3.当前遍历元素大于栈顶元素 收集结果,直到小于栈顶或栈为空加入栈while(!stack.isEmpty()&&temperatures[i]>temperatures[stack.peek()]){int index=stack.pop();res[index]=i-index;}stack.push(i);}}return res;}
}

496.下一个更大元素 I

题目链接

https://leetcode.cn/problems/next-greater-element-i/description/

解题思路

nums1是nums2的子集,所以记录nums1的map,key是nums1的值 value是nums1的索引下标
之后就和每日温度一样,求遍历元素右边第一个比它大的值,题目说了,每个元素的值都不一样,收集结果的时候判断当前栈弹出的值是否在map集合中,如果出现获取map集合的value就是nums1的索引下标记录结果。

code

class Solution {public int[] nextGreaterElement(int[] nums1, int[] nums2) {int[] res=new int[nums1.length];Arrays.fill(res,-1);Map<Integer,Integer> map=new HashMap<>();for(int i=0;i<nums1.length;i++){map.put(nums1[i],i);}System.out.println(map);Stack<Integer> stack=new Stack<>();stack.push(0);for(int i=1;i<nums2.length;i++){if(nums2[i]<=nums2[stack.peek()]){stack.push(i);}else{while(!stack.isEmpty()&&nums2[i]>nums2[stack.peek()]){int index=stack.pop();if(map.containsKey(nums2[index])){res[map.get(nums2[index])]=nums2[i];}}stack.push(i);}}return res;}
}

503.下一个更大元素II

题目链接

https://leetcode.cn/problems/next-greater-element-ii/description/

解题思路

理解环形数组,当前元素的下一个更大元素可能是它之前的元素,所以就要遍历俩圈就能收集到结果
遍历长度是 nums.length*2
向栈放入用 i%nums.length 下一轮遍历都会落到数组上
此题我注释了一部分,精简了代码,栈不为空,当前元素大于栈顶元素开始收集结果,其余情况(栈为空,当前元素小于等于栈顶元素)都入栈

code

class Solution {public int[] nextGreaterElements(int[] nums) {int[] res=new int[nums.length];Arrays.fill(res,-1);Stack<Integer> stack=new Stack<>();stack.push(0);//环形数组,最多遍历俩遍数组 索引for(int i=1;i<nums.length*2;i++){int index=i%nums.length;// if(nums[index]<=nums[stack.peek()]){//     stack.push(index);// }else {while(!stack.isEmpty()&&nums[index]>nums[stack.peek()]){int _index=stack.pop();res[_index]=nums[index];}stack.push(index);//  }}return res;}
}

这篇关于代码随想录 day 48 单调栈的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1091874

相关文章

Python实例题之pygame开发打飞机游戏实例代码

《Python实例题之pygame开发打飞机游戏实例代码》对于python的学习者,能够写出一个飞机大战的程序代码,是不是感觉到非常的开心,:本文主要介绍Python实例题之pygame开发打飞机... 目录题目pygame-aircraft-game使用 Pygame 开发的打飞机游戏脚本代码解释初始化部

Java中Map.Entry()含义及方法使用代码

《Java中Map.Entry()含义及方法使用代码》:本文主要介绍Java中Map.Entry()含义及方法使用的相关资料,Map.Entry是Java中Map的静态内部接口,用于表示键值对,其... 目录前言 Map.Entry作用核心方法常见使用场景1. 遍历 Map 的所有键值对2. 直接修改 Ma

深入解析 Java Future 类及代码示例

《深入解析JavaFuture类及代码示例》JavaFuture是java.util.concurrent包中用于表示异步计算结果的核心接口,下面给大家介绍JavaFuture类及实例代码,感兴... 目录一、Future 类概述二、核心工作机制代码示例执行流程2. 状态机模型3. 核心方法解析行为总结:三

python获取cmd环境变量值的实现代码

《python获取cmd环境变量值的实现代码》:本文主要介绍在Python中获取命令行(cmd)环境变量的值,可以使用标准库中的os模块,需要的朋友可以参考下... 前言全局说明在执行py过程中,总要使用到系统环境变量一、说明1.1 环境:Windows 11 家庭版 24H2 26100.4061

pandas实现数据concat拼接的示例代码

《pandas实现数据concat拼接的示例代码》pandas.concat用于合并DataFrame或Series,本文主要介绍了pandas实现数据concat拼接的示例代码,具有一定的参考价值,... 目录语法示例:使用pandas.concat合并数据默认的concat:参数axis=0,join=

C#代码实现解析WTGPS和BD数据

《C#代码实现解析WTGPS和BD数据》在现代的导航与定位应用中,准确解析GPS和北斗(BD)等卫星定位数据至关重要,本文将使用C#语言实现解析WTGPS和BD数据,需要的可以了解下... 目录一、代码结构概览1. 核心解析方法2. 位置信息解析3. 经纬度转换方法4. 日期和时间戳解析5. 辅助方法二、L

Python使用Code2flow将代码转化为流程图的操作教程

《Python使用Code2flow将代码转化为流程图的操作教程》Code2flow是一款开源工具,能够将代码自动转换为流程图,该工具对于代码审查、调试和理解大型代码库非常有用,在这篇博客中,我们将深... 目录引言1nVflRA、为什么选择 Code2flow?2、安装 Code2flow3、基本功能演示

IIS 7.0 及更高版本中的 FTP 状态代码

《IIS7.0及更高版本中的FTP状态代码》本文介绍IIS7.0中的FTP状态代码,方便大家在使用iis中发现ftp的问题... 简介尝试使用 FTP 访问运行 Internet Information Services (IIS) 7.0 或更高版本的服务器上的内容时,IIS 将返回指示响应状态的数字代

MySQL 添加索引5种方式示例详解(实用sql代码)

《MySQL添加索引5种方式示例详解(实用sql代码)》在MySQL数据库中添加索引可以帮助提高查询性能,尤其是在数据量大的表中,下面给大家分享MySQL添加索引5种方式示例详解(实用sql代码),... 在mysql数据库中添加索引可以帮助提高查询性能,尤其是在数据量大的表中。索引可以在创建表时定义,也可

使用C#删除Excel表格中的重复行数据的代码详解

《使用C#删除Excel表格中的重复行数据的代码详解》重复行是指在Excel表格中完全相同的多行数据,删除这些重复行至关重要,因为它们不仅会干扰数据分析,还可能导致错误的决策和结论,所以本文给大家介绍... 目录简介使用工具C# 删除Excel工作表中的重复行语法工作原理实现代码C# 删除指定Excel单元