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

目录

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.mfbz.cn/a/871377.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

为什么越来越多的IT青年转行网络安全?

目前&#xff0c;我国互联网已经从爆发增长期进入平稳发展阶段&#xff0c;同时每年大量计算机相关专业的毕业生涌入就业市场&#xff0c;导致IT行业逐渐趋于饱和状态&#xff0c;甚至出现裁员现象&#xff0c;去年很多大厂都有裁员&#xff0c;不少程序员再就业成了难题。 面…

Cache地址相联映像

直接相联映像&#xff1a;硬件电路直接连接 全相联映像; 电路难于设计和实现&#xff0c;只适用于小容量Cache&#xff0c;冲突率低 组相联映像&#xff1a;直接相联与全相联的折中。 冲突率 &#xff08;高&#xff0c;中&#xff0c;低&#xff09; 电路复杂度 其他 直接…

VSCode配置ssh免密连接远程服务器

我配置了免密设置(Windows利用ssh免密码登录Linux)&#xff0c;git bash已经能够正常连接了&#xff0c;但是vscode还是不行&#xff0c;很奇怪。 VSCode报错信息&#xff1a; [17:55:50.360] SSH Resolver called for "ssh-remote106.52.2.19", attempt 5, (Recon…

一文了解 Vue3 的 nextTick 大致信息

nextTick 是 Vue 3 中用于完成数据绑定和 DOM 更新后执行的方法&#xff0c;非常有用&#xff0c;也是 Vue 的一道比较常见的面试题。 1. 基本用法 nextTick 是一个异步方法&#xff0c;它允许我们在下一个 DOM 更新后执行回调函数。当更改了响应式数据并需要在更新后的 DOM …

C/C++控制台贪吃蛇游戏的实现

&#x1f680;欢迎互三&#x1f449;&#xff1a;程序猿方梓燚 &#x1f48e;&#x1f48e; &#x1f680;关注博主&#xff0c;后期持续更新系列文章 &#x1f680;如果有错误感谢请大家批评指出&#xff0c;及时修改 &#x1f680;感谢大家点赞&#x1f44d;收藏⭐评论✍ 一、…

OpManager Plus简单说明以及在Linux下的安装

目录 1 简介2 安装2.1 Linux下安装 1 简介 OpManager Plus 属于ManageEngine&#xff0c;是一款商业软件。 ManageEngine OpManager是一款全面的网络监视软件&#xff0c;可为网络管理员提供集成控制台&#xff0c;用于管理路由器&#xff0c;防火墙&#xff0c;服务器&#x…

Datawhale AI 夏令营 第四期 AIGC Task3

活动简介 活动链接&#xff1a;Datawhale AI 夏令营&#xff08;第四期&#xff09; 以及AIGC里面的本次任务说明&#xff1a;Task 3 进阶上分-实战优化 这次任务呢&#xff0c;主要是对知识的一个讲解&#xff0c;包括ComfyUI工具的使用啊&#xff0c;以及LoRA的原理啊&…

ansible搭建+ansible常用模块

ansible搭建 管理机安装ansible,被管理节点必须打开ssh服务 1.管理机安装ansible yum -y install ansible 2.查看版本 ansible --version ansible 2.9.27 3.查找配置文件 find /etc/ -name "*ansible*" /etc/ansible /etc/ansible/ansible.cfg 4.三台被管理机…

在Windows上配置VSCode MinGW+CMake(包括C++多线程编程的两套API:posix和win32)

创建目录 首先&#xff0c;需要电脑上安装VSCode, 并且创建三个文件夹&#xff1a;cmake、MinGW-posix、MinGW-w32 文件下载 下载posix-seh posix和win32分别是c多线程变成的两套API,可根据不同需求安装&#xff0c;现在先下载配置环境需要的几个文件 百度搜索MinGW-64 点…

使用JavaScript解决reCAPTCHA:完整教程

虽然reCAPTCHA有效地保护了网络内容&#xff0c;但有时它也会妨碍合法活动&#xff0c;例如研究、数据分析或其他与合规相关的自动化任务&#xff0c;这些任务需要与网络服务进行交互。 你将学到什么 在本博客中&#xff0c;我们将带你逐步了解如何使用JavaScript解决reCAPTC…

C++:stack类(vector和list优缺点、deque)

目录 前言 数据结构 deque vector和list的优缺点 push pop top size empty 完整代码 前言 stack类就是数据结构中的栈 C数据结构&#xff1a;栈-CSDN博客 stack类所拥有的函数相比与string、vector和list类都少很多&#xff0c;这是因为栈这个数据结构是后进先出的…

[CSCCTF 2019 Qual]FlaskLight1

打开题目 右键查看一下源代码 看到提示&#xff0c;需要用GET方search函数

g6解决拓扑图中dagre布局需要增加同级节点的问题(旁挂层同级节点相连)

背景&#xff1a;dagre可以在节点数据中配置layer字段&#xff0c;为节点指定层级&#xff0c;但layer的指定不能违背图结构与层次布局的原则&#xff0c;也就是说每一条边的起点的layer一定小于终点的layer值&#xff0c;否则会导致布局失败。 解决办法&#xff1a;动态添加节…

嵌入式人工智能ESP32(4-PWM呼吸灯)

1、PWM基本原理 PWM&#xff08;Pulse-width modulation&#xff09;是脉冲宽度调制的缩写。脉冲宽度调制是一种模拟信号电平数字编码方法。脉冲宽度调制PWM是通过将有效的电信号分散成离散形式从而来降低电信号所传递的平均功率的一种方式。所以根据面积等效法则&#xff0c;…

超简单亿图图示安装教程/快速入门指南及快捷键大全

一、软件介绍 Edraw Max&#xff08;亿图图示&#xff09;作为一款全类型的图形图表设计软件&#xff0c;深受广大用户的欢迎。目前&#xff0c;Edraw Max&#xff08;亿图图示&#xff09;里拥有20000多个符号&#xff0c;有效地满足使用者的需求&#xff1b;另外&#xff0c;…

JDBC基础Demo

pom.xml <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0" xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation"http://maven.apache.org/POM/4.0.0 …

STL六大组件

STL&#xff08;Standard Template Library&#xff0c;标准模板库&#xff09;是C标准库的一部分&#xff0c;提供了丰富且高效的数据结构和算法。STL主要由6大组件构成&#xff0c;分别是容器、算法、迭代器、适配器、仿函数和空间配置器。 容器&#xff08;Containers&#…

ES6 (一)——ES6 简介及环境搭建

目录 简介 环境搭建 可以在 Node.js 环境中运行 ES6 webpack 入口 (entry) loader 插件 (plugins) 利用 webpack 搭建应用 gulp 如何使用&#xff1f; 简介 ES6&#xff0c; 全称 ECMAScript 6.0 &#xff0c;是 JavaScript 的下一个版本标准&#xff0c;2015.06 发版…

ICC2:insertion delay会拉长同一skew group其他sink吗?

我正在「拾陆楼」和朋友们讨论有趣的话题,你⼀起来吧? 拾陆楼知识星球入口 来自知识星球提问: 已知一个skew group包含若干sink,针对其中一个sink设置insertion delay,希望工具把它做长,命令如下: create_clock_skew_group -clock xx -objects {xx xx} -name sg set_cl…

2024 江苏省第二届数据安全技术应用职业技能竞赛 初赛 部分wp

文章目录 一、前言二、参考文章三、题目&#xff08;解析&#xff09;数据安全解题赛1、ds_0602&#xff08;30分&#xff09;2、333.file&#xff08;45分&#xff09;3、pf文件分析&#xff08;35分&#xff09;4、丢失的资料&#xff08;45分&#xff09;5、greatphp&#x…