代码随想录算法训练营day48:单调栈

news2025/1/4 15:27:39

目录

739. 每日温度

503.下一个更大元素II

分析:

42. 接雨水

本质:

暴力解法 分析:

双指针优化

单调栈

84.柱状图中最大的矩形

分析:

双指针:

单调栈


739. 每日温度

力扣题目链接(opens new window)

请根据每日 气温 列表,重新生成一个列表。对应位置的输出为:要想观测到更高的气温,至少需要等待的天数。如果气温在这之后都不会升高,请在该位置用 0 来代替。

例如,给定一个列表 temperatures = [73, 74, 75, 71, 69, 72, 76, 73],你的输出应该是 [1, 1, 4, 2, 1, 1, 0, 0]。

提示:气温 列表长度的范围是 [1, 30000]。每个气温的值的均为华氏度,都是在 [30, 100] 范围内的整数。

分析:

此时有同学可能就疑惑了,那result[6] , result[7]怎么没更新啊,元素也一直在栈里。

其实定义result数组的时候,就应该直接初始化为0,如果result没有更新,说明这个元素右面没有更大的了,也就是为0。

注意~~:储存下标时,写代码很容易写错!要注意

int* dailyTemperatures(int* temperatures, int temperaturesSize, int* returnSize) {
    int s[temperaturesSize];//记录的是温度的下标
    int top=-1;

    for(int i=0;i<temperaturesSize;i++){
        if(top==-1 || temperatures[s[top]]>=temperatures[i]) s[++top]=i;
        else {
            while(top!=-1 && temperatures[s[top]]<temperatures[i]){
                int x=s[top];
                
                temperatures[x]=i-x;
                top--;
            }
            s[++top]=i;
        }
    }

    while(top!=-1){
        temperatures[s[top]]=0;
        top--;
    }
    *returnSize=temperaturesSize;
    return temperatures;
}

503.下一个更大元素II

力扣题目链接(opens new window)

给定一个循环数组(最后一个元素的下一个元素是数组的第一个元素),输出每个元素的下一个更大元素。数字 x 的下一个更大的元素是按数组遍历顺序,这个数字之后的第一个比它更大的数,这意味着你应该循环地搜索它的下一个更大的数。如果不存在,则输出 -1。

示例 1:

  • 输入: [1,2,1]
  • 输出: [2,-1,2]
  • 解释: 第一个 1 的下一个更大的数是 2;数字 2 找不到下一个更大的数;第二个 1 的下一个最大的数需要循环搜索,结果也是 2。

提示:

  • 1 <= nums.length <= 10^4
  • -10^9 <= nums[i] <= 10^9

分析:

思路:

相当于循环两遍nums数组

取最大值的情况照旧

但对于第二轮循环时,有些元素已经完成计算,此时不需要计算覆盖情况,continue掉

runtime error: load of null pointer of type ‘int‘

该题需要return一个数组,而在不申请空间(不malloc)的情况下函数内建立的数组是局部变量,无法带回主函数。

所以在开辟需要返回的数组空间时,在leetcode里需要用malloc

复杂度:o(n),o(n)

int* nextGreaterElements(int* nums, int numsSize, int* returnSize) {
    int index[2*numsSize];//index存放的数值,第二轮会比第一轮大numssize
    int top=-1;
    int *ans;
    ans=(int*)malloc(sizeof(int)*2*numsSize);
    for (int i=0;i<numsSize;i++){
        ans[i]=-1;//初始化为-1,因为没找到最大的数,就会赋值为-1;
    }
    //i在0~2n,ans、nums要求0~n,index要求0~2n

    for(int i=0;i<numsSize*2;i++){//如果存在下一个更大的数,循环两轮一定能找到
        //进栈的情况
        if(top == -1 || nums[index[top]%numsSize] >= nums[i%numsSize]) {
            index[++top]=i;
        }

        //出栈;是否要计算
        else {
            while(top!=-1 && nums[index[top]%numsSize] < nums[i%numsSize]){
                int x = index[top];
                if(ans[x%numsSize]!=-1) {
                    top--;
                    continue;
                }
                ans[x%numsSize]=nums[i%numsSize]; 
                top--;
            }
            index[++top]=i;
        }
    }
    *returnSize=numsSize;
    return ans;
}

42. 接雨水

力扣题目链接(opens new window)

给定 n 个非负整数表示每个宽度为 1 的柱子的高度图,计算按此排列的柱子,下雨之后能接多少雨水。

示例 1:

  • 输入:height = [0,1,0,2,1,0,1,3,2,1,2,1]
  • 输出:6
  • 解释:上面是由数组 [0,1,0,2,1,0,1,3,2,1,2,1] 表示的高度图,在这种情况下,可以接 6 个单位的雨水(蓝色部分表示雨水)。

本质:

这种类型的题目可以从两个方向讨论:

按列看:

左边最高的柱子,右边最高的柱子,会围成一个凹槽,可以放水

每一列的雨量=min(两边最高)-本列高度

按行看:

按照一个个凹槽来求

暴力解法 分析:

如果按照列来计算的话,宽度一定是1了,我们再把每一列的雨水的高度求出来就可以了。

可以看出每一列雨水的高度,取决于,该列 左侧最高的柱子和右侧最高的柱子中最矮的那个柱子的高度。

可以看出每一列雨水的高度,取决于,该列 左侧最高的柱子和右侧最高的柱子中最矮的那个柱子的高度。

这句话可以有点绕,来举一个理解,例如求列4的雨水高度,如图:

42.接雨水3

列4 左侧最高的柱子是列3,高度为2(以下用lHeight表示)。

列4 右侧最高的柱子是列7,高度为3(以下用rHeight表示)。

列4 柱子的高度为1(以下用height表示)

那么列4的雨水高度为 列3和列7的高度最小值减列4高度,即: min(lHeight, rHeight) - height。

列4的雨水高度求出来了,宽度为1,相乘就是列4的雨水体积了。

此时求出了列4的雨水体积。

一样的方法,只要从头遍历一遍所有的列,然后求出每一列雨水的体积,相加之后就是总雨水的体积了。

首先从头遍历所有的列,并且要注意第一个柱子和最后一个柱子不接雨水,代码如下:

for (int i = 0; i < height.size(); i++) {
    // 第一个柱子和最后一个柱子不接雨水
    if (i == 0 || i == height.size() - 1) continue;
}

在for循环中求左右两边最高柱子,代码如下:

int rHeight = height[i]; // 记录右边柱子的最高高度
int lHeight = height[i]; // 记录左边柱子的最高高度
for (int r = i + 1; r < height.size(); r++) {
    if (height[r] > rHeight) rHeight = height[r];
}
for (int l = i - 1; l >= 0; l--) {
    if (height[l] > lHeight) lHeight = height[l];
}

最后,计算该列的雨水高度,代码如下:

int h = min(lHeight, rHeight) - height[i];
if (h > 0) sum += h; // 注意只有h大于零的时候,在统计到总和中

时间复杂度为O(n^2),空间复杂度为O(1)——超时 

双指针优化

区别于单调栈问题!!单调栈找的是附近的最大值、最小值;如果按照列求,实际上找的是左边\右边 的最大值,而不是最近的大值

在暴力解法中,我们可以看到只要记录左边柱子的最高高度 和 右边柱子的最高高度,就可以计算当前位置的雨水面积,这就是通过列来计算。

当前列雨水面积:min(左边柱子的最高高度,记录右边柱子的最高高度) - 当前柱子高度。

为了得到两边的最高高度,使用了双指针来遍历,每到一个柱子都向两边遍历一遍,这其实是有重复计算的。我们把每一个位置的左边最高高度记录在一个数组上(maxLeft),右边最高高度记录在一个数组上(maxRight),这样就避免了重复计算。

当前位置,左边的最高高度是前一个位置的左边最高高度和本高度的最大值。

即从左向右遍历:maxLeft[i] = max(height[i], maxLeft[i - 1]);

从右向左遍历:maxRight[i] = max(height[i], maxRight[i + 1]);

代码如下:

class Solution {
public:
    int trap(vector<int>& height) {
        if (height.size() <= 2) return 0;
        vector<int> maxLeft(height.size(), 0);
        vector<int> maxRight(height.size(), 0);
        int size = maxRight.size();

        // 记录每个柱子左边柱子最大高度
        maxLeft[0] = height[0];
        for (int i = 1; i < size; i++) {
            maxLeft[i] = max(height[i], maxLeft[i - 1]);
        }
        // 记录每个柱子右边柱子最大高度
        maxRight[size - 1] = height[size - 1];
        for (int i = size - 2; i >= 0; i--) {
            maxRight[i] = max(height[i], maxRight[i + 1]);
        }
        // 求和
        int sum = 0;
        for (int i = 0; i < size; i++) {
            int count = min(maxLeft[i], maxRight[i]) - height[i];
            if (count > 0) sum += count;
        }
        return sum;
    }
};

单调栈

应用范围:通常是一维数组,要寻找任一个元素的右边或者左边第一个比自己大或者小的元素的位置,此时我们就要想到可以用单调栈了。

而接雨水这道题目,我们正需要寻找一个元素,右边最大元素以及左边最大元素,来计算雨水面积。

最关键:直接用单调栈,通过栈顶、栈顶的下一个元素、即将入栈元素,三个为一组,计算盛的雨水

int trap(int* height, int heightSize) {
    int sum=0;

    int s[heightSize];//存放location
    int top=-1;
    for(int i=0;i<heightSize;i++){
        if(top==-1 || height[s[top]] > height[i]) s[++top]=i;
        else if(height[s[top]] == height[i]) s[top]=i;
        else {
            while(top!=-1 && height[s[top]] < height[i]){
                int mid= s[top--];
                if(top>=0){
                    int left=s[top];
                    sum += fmin(height[i]-height[mid],height[left]-height[mid])*(i-left-1);
                    printf("left=%d mid=%d right=%d sum=%d \n",left,mid,i,sum);
                }
            }
            s[++top]=i;
        }
    }
    return sum;
}

84.柱状图中最大的矩形

力扣题目链接(opens new window)

给定 n 个非负整数,用来表示柱状图中各个柱子的高度。每个柱子彼此相邻,且宽度为 1 。

求在该柱状图中,能够勾勒出来的矩形的最大面积。

分析:

对于每个柱子本体,考虑最终矩形高度为自身高度的情况

找左边第一个更矮的柱子,找右边第一个 更矮的柱子——最宽能宽到哪里

同理也可以用暴力算法、双指针、单调栈

双指针:

记录每个柱子 左边\右边第一个小于该柱子的下标

和接雨水有区别

因为找左边\右边第一个小于该柱子的下标,而不是左边最大的

如果要一个个遍历,那么就是暴力法了

既然开辟了数组,那就利用数组来简化

分为两种情况讨论:

1、找左边第一个比本体矮的:

最外层从左到右循环,j从i-1开始往前,当hj比hi大时——跳到hj左边第一个比j矮的 即leftmin[j]

hj比hi小——找到了,就是j

2、右边:

从右到左循环!左侧的需要建立在右侧已经找到的基础上

int largestRectangleArea(int* heights, int heightsSize) {
    int leftmin[heightsSize];
    int rightmin[heightsSize];
    for (int i=0;i<heightsSize;i++){
        leftmin[i]=-1;
        rightmin[i]=heightsSize;
    }

    for(int i=1;i<heightsSize;i++){
        int j=i-1;
        while(j>=0){
            if(heights[j] < heights[i]){
                leftmin[i]=j;
                break;
            }
            j=leftmin[j];
        }
    }

    for(int i=heightsSize-2;i>=0;i--){
        int k=i+1;
        while(k<heightsSize){
            if(heights[k] <heights[i] ){
                rightmin[i]=k;
                break;
            }
            k=rightmin[k];
        }
    }

    int ansmax=0;
    for(int i=0;i<heightsSize;i++){
        int h=heights[i];
        int l=rightmin[i]-leftmin[i]-1;
        ansmax=fmax(ansmax, l*h);
        printf("%d %d %d \n",leftmin[i],rightmin[i],ansmax);

    }
    return ansmax;

    
}

单调栈

本来是想找两个栈分别求本位置 左边\右边第一个小于的元素

后面发现当栈里从大到小的顺序,才能保证栈顶元素找到左右两边第一个小于栈顶元素的柱子。栈顶和栈顶的下一个元素以及要入栈的三个元素组成了我们要求最大面积的高度和宽度。

开头和结尾放0来便于计算! 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2055709.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

这个大佬一年连中五篇顶会!

1、SLAM/3DGS/三维点云/医疗图像/扩散模型/结构光/Transformer/CNN/Mamba/位姿估计 顶会论文指导 2、基于扩散模型的跨域鲁棒自动驾驶场景理解 3、基于环境信息的定位&#xff0c;重建与场景理解 4、轻量级高保真Gaussian Splatting 5、基于大模型与GS的 6D pose estimatio…

【Redis】Redis线程与IO模型—(三)

Redis线程与IO模型 一、Redis 单线程二、多路复用机制三、Redis 6.0 多线程特性四、IO 多线程配置 一、Redis 单线程 通常说 Redis 是单线程&#xff0c;主要是指 Redis 的网络 IO 和键值对读写是由一个线程来完成的&#xff0c;其他功能&#xff0c;比如持久化、异步删除、集…

软件测试之全面质量管理

一.什么是TQM&#xff1f; 全面质量管理 英文&#xff1a;Total Quality Management TQM可以被定义为一种管理技术&#xff0c;用于改进与产品相关的过程、产品、服务和其他方法。 它关注的是整个业务&#xff0c;而不仅仅是一个特定的项目或过程。 二.TQM原则 以顾客为关注…

软考作弊率下降了78.68%!官方为了防止作弊做出了哪些努力?

01\软考违纪违规人数对比 2024年上半年软考&#xff08;机考&#xff09;共有52名考生被判违纪违规行为&#xff0c;其中浙江考区有9人&#xff0c;山东枣庄考区有10人&#xff0c;江苏考区有33人。 2023年下半年软考&#xff08;机考&#xff09;共有7名考生被判违纪违规行为…

磁盘格式化文件恢复:一文看懂数据恢复操作

当你意识到关键的硬盘已经被格式化&#xff0c;而且你不能获取里面的内容时&#xff0c;这会是非常令人沮丧的。这种情况可能是因为硬盘被不小心格式化&#xff0c;或者是你在试图修正一些问题、调整文件系统或者释放存储空间时&#xff0c;有意进行的格式化。无论具体情况是什…

【论文学习与撰写】论文中公式的编辑,Mathtype的使用,全文编号排版,智能截图识别公式,公式编号自动更新

1、准备工作 在word中安装mathtype插件&#xff0c; 2、插入公式 在想要插入公式的地方&#xff0c;点击右编号&#xff0c;在里面输入公式&#xff0c;保存&#xff0c;关闭&#xff0c;就会得到插入的公式。 3、公式编号的编辑&#xff0c;公式编号自动更新 要想插入的公式…

笔试练习day5

目录 游游的you题目解析解法方法一贪心方法二 腐烂的苹果题目解析例子1例子2解法多源BFS最短路径代码代码解析 JZ62 孩子们的游戏(圆圈中最后剩下的数)题目解析解法方法一模拟环形链表模拟数组模拟 方法二递推/递归/动态规划状态表示状态转移方程代码 感谢各位大佬对我的支持,如…

CORS error 302 Found

CORS error && 302 Found 场景 单点登录认证&#xff1a;访问A系统&#xff0c;在B系统登录认证 此处代码为A系统 controller ResponseBodyGetMapping("/idp/loginCheck")public void loginCheck(HttpServletRequest request, HttpServletResponse httpR…

基于vue框架的爱学习分享平台ud317(程序+源码+数据库+调试部署+开发环境)系统界面在最后面。

系统程序文件列表 项目功能&#xff1a;用户,学科分类,交流答疑,论坛交流,学习资料 开题报告内容 基于Vue框架的爱学习分享平台 开题报告 一、项目背景与意义 随着互联网技术的飞速发展&#xff0c;知识的获取与传播方式正经历着前所未有的变革。在线教育平台逐渐成为满足…

【独立站搭建经验分享】B2C独立站如何搭建?怎么推广?

如果你的产品有C端属性&#xff0c;可能你就需要考虑建一个B2C独立站&#xff0c;用来满足访客的浏览和在线下单。那么B2C独立站应该怎么搭建&#xff0c;选择什么推广方式&#xff0c;这个前期最好有一些基本了解&#xff0c;本篇内容可以针对你最关心的问题进行讲解&#xff…

【文献】3D Gaussian Splatting for Real-Time Radiance Field Rendering

论文地址&#xff1a;https://arxiv.org/abs/2308.04079 项目&#xff1a; https://repo-sam.inria.fr/fungraph/3d-gaussian-splatting/ 代码&#xff1a; git clone https://github.com/graphdeco-inria/gaussian-splatting —recursive 一、文章概述 1.问题导向 辐射场…

《笑谈设计模式》 — 23种尝鲜版(未完待续......

引子&#xff1a;无论在平时开发过程中&#xff0c;还是深夜研读源码亦或者面试时都对遇到——设计模式。比如说Spring中的单例模式&#xff08;bean单例&#xff09;、工厂模式&#xff08;bean创建&#xff09;、代理模式&#xff08;动态代理&#xff09;、策略模式等。我们…

RTA-VRTE适配Orin

RTA-VRTE适配Orin sudo minicom -w -D /dev/ttyACM0 用户名 nvidia密码 123456底下的的rj45 ,对应的是eqos_0, (本次porting使用该接口)底下的的DEBUG(USB Micro-B)串口连接(本次porting使用该接口)上侧边的rj45,对应的是mgbe0_0#设置Orin的IP和默认网卡eqos_0

C语言:一维、二维数组详解

目录 一、数组的概念 二、一维数组创建和初始化 2.1 数组创建 2.2 数组的初始化 2.3 数组的类型 三、一维数组的使用 3.1 数组的下标 3.2 数组的输入与输出 四、⼀维数组在内存中的存储 五、sizeof计算数组元素个数 六、二维数组 6.1 二维数组的概念 6.2 二维数组…

虚幻游戏开发 | shader性能优化

如何查看shader性能消耗 1.Alt 8切换Shader Complexity mode查看shader性能消耗 (Alt 4切换回Lit Mode正常光照模式) 2.材质蓝图里查看instructions数量&#xff08;如图中箭头所示&#xff09; 3.直接在target platform上测试是最准确的。 &#xff08;running fewer instru…

思特科技案例:北京欢乐谷光影乐园

01      在北京欢乐谷的亲子领域&#xff0c;藏着一处“面积近400平米&#xff0c;炫酷堪比魔法世界的、美轮美奂的光影空间&#xff0c;做到了“让娃来了不想走&#xff0c;一玩就是一下午”。    思特科技案例&#xff1a;北京欢乐谷光影乐园      02      作…

去抖音视频水印的软件免费推荐,精选四款实用工具助你轻松去除水印

随着抖音等短视频平台的流行&#xff0c;我们常常会遇到想要保存某些视频但不想保留水印的情况。本文将为您推荐四款去抖音视频水印的实用工具&#xff0c;帮助您轻松去除视频中的水印&#xff0c;保存清晰的内容。 工具一&#xff1a;奈斯水印助手(小程序) 推荐指数&#xf…

DMDSC学习

DMDSC学习 ​ DM 共享存储数据库集群&#xff0c;允许多个数据库实例同时访问、操作同一数据库&#xff0c;具有高可用、高性能、负载均衡等特性。DMDSC 支持故障自动切换和故障自动重加入&#xff0c;某一个数据库实例故障后&#xff0c;不会导致数据库服务无法提供。 DMDSC…

前端使用miniO上传文件

项目背景:vue2&#xff0c;前提是请先安装miniO,若安装引入时报错&#xff0c;那就是版本不对&#xff0c;通常指定版本安装即可。 页面样式&#xff1a; 前端vue页面代码&#xff1a; //<el-form>表单中:<el-form-item label"文件" prop"fileIds&q…

链表OJ题——反转一个单链表

文章目录 一、题目链接二、解题思路三、解题代码 一、题目链接 原地翻转链表 二、解题思路 原地翻转链表&#xff0c;空间复杂度为O(1)&#xff0c;时间复杂度O(n) 三、解题代码