[ 一刷完结撒花!! ] Day50 力扣单调栈 : 503.下一个更大元素II |42. 接雨水 | 84.柱状图中最大的矩形

news2025/1/10 16:46:20

Day50 力扣单调栈 : 503.下一个更大元素II |42. 接雨水 | 84.柱状图中最大的矩形

  • 503.下一个更大元素II
    • 第一印象
    • 看完题解的思路
    • 实现中的困难
    • 感悟
    • 代码
  • 42. 接雨水
    • 第一印象
    • 看完题解的思路
      • 暴力解法
      • 单调栈解法
    • 实现中的困难
    • 感悟
    • 代码
  • 84.柱状图中最大的矩形
    • 第一印象
    • 看完题解的思路
    • 感悟
    • 代码

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

第一印象

好, 那我就自己试试.

做出来了.

之前的单调栈题目就是遍历一遍数组, 找这一遍里右面最大的那个.

这道题说是循环数组, 其实也就是找两圈这个数组.

比如 5 2 3. 5是比3大的, 第一圈里只能找出3比2大, 第二圈才能找到5比3大.

有两个方式我觉得, 一个是找两圈数组, 也是我下面代码写的.

或者直接改造这个数组, 变成自己的两倍大小, 元素是自己的两遍, 比如5 2 3 5 2 3.

我自己写的代码,请看vcr:

class Solution {
    public int[] nextGreaterElements(int[] nums) {
        int[] result = new int[nums.length];
        Deque<Integer> stack = new LinkedList<>();
        int loop = 1;

        Arrays.fill(result, -1);
        stack.push(0);

        for (int i = 1; i < nums.length; i++) {
            if (nums[i] <= nums[stack.peek()]) {
                stack.push(i);
            } else {
                while (!stack.isEmpty() && nums[i] > nums[stack.peek()]) {
                    //弹出栈顶元素,是下标
                    int index = stack.pop();
                    //记录结果
                    result[index] = nums[i];
                }
                stack.push(i);
            }
            if (loop == 1 && i == nums.length - 1) {
                //因为循环结束 i++, 而第二圈应该从 i=0 开始.
                i = -1;
                loop--;
                continue;
            }
        }

        return result;
    }
}

看完题解的思路

确实, 就是我上面说的两种方式, 但是扩大数组的复杂度就是O(n)了

而走两圈的方式, 卡哥比我做的要聪明一些, 用 % .

其实我也想到了, 就是感觉容易混乱在数学里.

for(int i = 0; i < 2*size; i++) {
    while(!st.empty() && nums[i % size] > nums[st.peek()]) {
        result[st.peek()] = nums[i % size];//更新result
        st.pop();//弹出栈顶
    }
    st.push(i % size);
}

实现中的困难

在我的代码里

if (loop == 1 && i == nums.length - 1) {
    //因为循环结束 i++, 而第二圈应该从 i=0 开始.
    i = -1;
    loop--;
    continue;
}

走两圈的逻辑, 应该让 i = -1 ,而不是 i = 0

因为for循环结束之后会有 i++ .

感悟

虽然麻烦了点, 但我还是做对了

代码

上面给出了.

42. 接雨水

接雨水这道题目是 面试中特别高频的一道题,也是单调栈 应用的题目,大家好好做做。

建议是掌握 双指针 和单调栈,因为在面试中 写出单调栈可能 有点难度,但双指针思路更直接一些。

在时间紧张的情况有,能写出双指针法也是不错的,然后可以和面试官在慢慢讨论如何优化。
https://programmercarl.com/0042.%E6%8E%A5%E9%9B%A8%E6%B0%B4.html

第一印象

想了一下, 感觉情况比较多

  • 两边都比自己高, 选小的那个 - 自己 = 水
  • 两边有一边比自己高, ??? 这种情况咋办呢
  • 两边都没自己高, 没有水

看题解!

看完题解的思路

暴力解法

找到右边第一个比自己大的, 左面第一个比自己大的. 就能算出来自己这里有多少水. 硬暴力去算.

单调栈解法

对于单调栈, 前面的题目已经会了找右边更大的(栈元素递增), 右边更小的(栈元素递减).

那怎么找左边更大的呢? 我第一反应是逆序遍历数组.

可以, 但有更好的方式. 这道题就是找到三个柱子, 左边更大+自己+右边更大

自己就是栈顶元素, 拿来的元素 i 可能是右边更大的(不更大就push入栈了), 那左边更大的在哪?

其实就是栈顶下面的那个元素, 如图:

因为栈里是递增的, 我自己下面那个就是 我左面第一个更大的呀

太tm巧妙了!!!
在这里插入图片描述

看完了, 会了

重要的是, 这道题是横向的求雨水面积, 我一直想的是 纵向的求雨水面积.

我听完题解还是觉得纵向的求没有错

但是细看

在这里插入图片描述

这个地方左面第一个更高和右边第一个更高最小的是 1

自己是 0

那么纵向雨水就是 1 , 算不到红色方块那里的

实现中的困难

思路清晰就不难实现

感悟

确实比较难

一个是纵向和横向计算容易糊涂

我也觉得横向计算的话, 下图的两个地方不会重复吗?

在这里插入图片描述

不会的

因为第二个 1 拿来的时候, 就比 0大了, 就会进行一次计算.

算出雨水量 1, 并把 0 pop掉, 第二个 1 push进去

拿来 3 的时候, 比栈顶的第二个 1 更大, 就会进行一次计算.

但是栈顶下面的元素就是第一个 1 , 在计算高度的时候, 取第一个 1 和 3 更小的那个 再减去栈顶的第二个 1 . 高度就是0 , 就白计算了.

然后第二个 1 pop

这个时候栈顶是第一个 1 , 栈顶下面的元素是 2 , 就会进行一次计算.

算出 高度 1 ,宽度 3, 雨水量 3.

然后pop掉第一个 1, 3 比栈顶的 2 还要大. 但是呢, pop 2 之后, 栈就空了, 就不会计算, 只是 pop 了2.

最后 3 push进栈, 再去算后面的元素.

顿悟 咯~~~~~~~~~~~~~

代码

class Solution {
   public int trap(int[] height) {
       int result = 0;
       Deque<Integer> stack = new LinkedList<>();

       stack.push(0);
       for (int i = 1; i < height.length; i++) {
           if (height[i] < height[stack.peek()]) {
               stack.push(i);
           } else if (height[i] == height[stack.peek()]) {
               stack.pop();
               stack.push(i);
           } else {
               while (!stack.isEmpty() && height[i] > height[stack.peek()]) {
                   int curIndex = stack.pop();
                   int cur = height[curIndex];
                   //如果还有左边的话
                   if (!stack.isEmpty()) {
                       int min = Math.min(height[i], height[stack.peek()]);
                       //求高度
                       int h = min - cur;
                       int w = i - stack.peek() - 1;
                       //我纵向的求
                       int rain = w * h;
                       result += rain;
                   }
               }
               stack.push(i);
           }
       }
       return result;
   }
}

84.柱状图中最大的矩形

https://programmercarl.com/0084.%E6%9F%B1%E7%8A%B6%E5%9B%BE%E4%B8%AD%E6%9C%80%E5%A4%A7%E7%9A%84%E7%9F%A9%E5%BD%A2.html

今天是训练营最后一天,恭喜坚持两个月的录友们,接下来可以写一篇自己
代码随想录一刷的总结。好好回顾一下,这两个月自己的博客内容,以及自己的收获。

第一印象

我也觉得是要找左右两边第一个更小的

题解也是这么写的.

但是为啥呢

我只是有一种直觉, 它不像接雨水那样比较直观.

看完题解的思路

我明白了, 拿题里的例子说明一下

在这里插入图片描述

比如拿到元素 1 , 向左找第一个更小, 没有. 向右找第一个更小, 没有.

所以 1 的高度, 贯穿整个数组.

比如拿到 5, 向左找第一个更小是 1 . 向右找第一个更小, 是2.

那么 5 的高度就是从 1 贯穿到 2, 宽度其实就是 2 的下标 - 1 的下标 - 1.

这就是为什么要向左向右找第一个更小的原因.

也就是看现在这个元素(是高度)能贯穿多长(宽度).

然后这道题还有个处理技巧, 给数组扩容, 头和尾都加上高度 0 .

这样就方便算头和尾的元素的面积. 而且还能算出那个 最矮 的 1 的面积. 如图

在这里插入图片描述

但这里又会产生一个问题

对于相等的情况我自己写的时候的处理是, 什么都不做

if (heights[i] > heights[stack.peek()]) {
    stack.push(i);
} else if (heights[i] == heights[stack.peek()]) {
    //不处理, 相同的话先放进来的元素会让面积更大
} else {

比如 1 4 4 3.

肯定是第一个 4 算出的面积更大, 第二个 4 就是 4*1, 第一个 4 是 4 * 2.

所以遇到相同元素, 什么都不做就行了.

但是由于头加了 0 , 如果第一个高度还是 0的话就会出现错误.

在这里插入图片描述
扩容之后的数组是 0 0 9 0

因为相同元素什么都不干, 所以第二个 0 不会压入栈

在这里插入图片描述

这样计算 这个 9 的时候, 宽度就是 2 了. 就错了!!!

所以不能让相同元素什么都不干.

而是像接雨水一样, pop 在push 可以, 只push 也可以

还拿 1 4 4 3 的例子

pop 再 push, 栈顶就是第二个 4 , 下面是 1 . 拿到元素 3 的时候, 计算的宽度也是 3 - 0 - 1 = 2. 算的是到 1 到 3 的宽度. 是大的那个.

只push呢, 栈顶是第二个4 然后第一个4, 然后是 1 . 就是先算一个小的面积, 再算大的.

都是对的, 就是什么都不做不行.

感悟

这题也挺奇妙, 但是栈里递增递减不是什么难的.

主要就是头和尾 + 0 的小技巧

代码

class Solution {
    public int largestRectangleArea(int[] heights) {
        int maxArea = 0;
        Deque<Integer> stack = new LinkedList<>();

        // 数组扩容,在头和尾各加入一个元素
        int [] newHeights = new int[heights.length + 2];
        newHeights[0] = 0;
        newHeights[newHeights.length - 1] = 0;
        for (int index = 0; index < heights.length; index++){
            newHeights[index + 1] = heights[index];
        }

        heights = newHeights;

        stack.push(0);
        for (int i = 1; i < heights.length; i++) {
            if (heights[i] > heights[stack.peek()]) {
                stack.push(i);
            } else if (heights[i] == heights[stack.peek()]) {
                stack.pop();
                stack.push(i);
            } else {
                while (!stack.isEmpty() && heights[i] < heights[stack.peek()]) {
                    int curIndex = stack.pop();
                    int cur = heights[curIndex];

                    if (!stack.isEmpty()) {
                        int weight = i - stack.peek() - 1;
                        int area = cur * weight;
                        System.out.println(area + "=" + cur + "*" + weight);
                        maxArea = Math.max(maxArea, area);
                    }    
                }
                stack.push(i);
            }
        }

        return maxArea;
    }
}

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

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

相关文章

037、目标检测-SSD实现

之——简单实现 目录 之——简单实现 杂谈 正文 1.类别预测层 2.边界框预测 3.多尺度输出联结做预测&#xff08;提高预测效率&#xff09; 4.多尺度实现 5.基本网络块 6.完整模型 杂谈 原理查看&#xff1a;037、目标检测-算法速览-CSDN博客 正文 1.类别预测层 类别…

【力扣面试经典150题】(链表)K 个一组翻转链表

题目描述 力扣原文链接 给你链表的头节点 head &#xff0c;每 k 个节点一组进行翻转&#xff0c;请你返回修改后的链表。 k 是一个正整数&#xff0c;它的值小于或等于链表的长度。如果节点总数不是 k 的整数倍&#xff0c;那么请将最后剩余的节点保持原有顺序。 你不能只…

“释放视频潜力,批量放大视频尺寸,高效提升视频质量“

在视频制作和编辑的过程中&#xff0c;我们经常需要调整视频的尺寸。然而&#xff0c;一个一个地手动调整不仅耗时&#xff0c;还容易出错。为了解决这个问题&#xff0c;现在有一款全新的视频批量剪辑工具&#xff0c;可以帮助你批量将视频尺寸放大&#xff0c;提升工作效率。…

Argo Rollouts结合Service进行Blue-Green部署

删除03 部署04 rootk8s-master01:~/learning-jenkins-cicd/09-argocd-and-rollout/rollout-demos# kubectl delete -f 03-rollouts-with-prometheus-analysis.yaml rootk8s-master01:~/learning-jenkins-cicd/09-argocd-and-rollout/rollout-demos# kubectl apply -f 04-rol…

C++多线程编程(3):接收线程处理函数的返回值

文章首发于我的个人博客&#xff1a;欢迎大佬们来逛逛 文章目录 处理带返回值的函数asyncpackaged_taskpromise 处理带返回值的函数 有三种方法&#xff1a; asyncpackaged_taskpromise async 第一种方法是使用 async 函数。 步骤&#xff1a; 使用 async 创建线程处理函…

pom.xml格式化快捷键

在软件开发和编程领域&#xff0c;"格式化"通常指的是将代码按照一定的规范和风格进行排列&#xff0c;以提高代码的可读性和维护性。格式化代码有助于使代码结构清晰、统一&#xff0c;并符合特定的编码规范。 格式化可以包括以下方面&#xff1a; 缩进&#xff1a…

HR应用在线人才测评,给企业招聘带来的好处

一、什么是人才测评&#xff1f; 人才测评是指运用一系列的科学方法&#xff0c;对人的基本素质&#xff0c;专业能力&#xff0c;心理健康&#xff0c;性格进行选拔&#xff0c;评价及发展人才的一种科学方法。近十多年&#xff0c;它被广泛运用于国有大型企业的人才招聘和人…

策略模式在数据接收和发送场景的应用(升级版)

1.背景 在数据接收和发送场景打算使用了 if else 进行判断&#xff1a; if("A".equals(system)){ASystem.sync("向A同步数据"); } if("B".equals(system)){BSystem.sync("向B同步数据"); } ... 非常麻烦&#xff0c;需求多了很臃肿&…

AI对开发者职业的影响,保持领先的7 个行动指南

在不断发展的技术领域&#xff0c;人工智能(AI)已经成为一股变革性的力量&#xff0c;重塑了行业&#xff0c;重新定义了我们解决问题的方式。对于开发人员来说&#xff0c;学习AI的决定不仅仅是为了保持相关性&#xff0c;而是在他们的职业生涯中开启一个新的可能性维度。 1.…

Scalable Exact Inference in Multi-Output Gaussian Processes

Orthogonal Instantaneous Linear Mixing Model TY are m-dimensional summaries&#xff0c;ILMM means ‘Instantaneous Linear Mixing Model’&#xff0c;OILMM means ‘Orthogonal Instantaneous Linear Mixing Model’ 辅助信息 作者未提供代码

【接口测试】最细Fiddle抓包辅助接口实战,抓包全过程总结...

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 1、为什么需要抓包…

Python开源项目周排行 2023年第38周

#2023年第38周2023年11月19日1easybc用于解析分组加密算法的输入 [EasyDC] 程序&#xff0c;然后根据差分密码分析对分组加密算法进行安全分析。它支持以下功能&#xff1a; EasyBC 的解释器。 基于 SMT 的方法&#xff0c;用于确定各种密码操作的分支数。 S-box 中的差分传播建…

C++多线程编程(1):线程的创建方式

文章首发于我的个人博客&#xff1a;欢迎大佬们来逛逛 文章目录 进行与线程C中如何实现多线程创建线程的多种方式无参函数lambda表达式常成员函数not常成员引用函数智能指针仿函数类的普通成员函数综合测试 进行与线程 多线程是指多个线程并发执行的过程。 进程与线程的关系&…

WIFI版本云音响设置教程腾讯云平台版本

文章目录 WIFI版本云音响设置教程腾讯云平台版本一、申请设备三元素1.腾讯云物联网平台2.创建产品3.设置产品参数4.添加设备5.获取三元素 二、设置设备三元素1.打开MQTTConfigTools2.计算MQTT参数3.使用windows电脑的WIFI连接到设备热点4.设置参数 三、腾讯云物联网套件协议使用…

Linux基础全整理 从入门到放弃,一些想说的话

阅读目录 断更后一些想说的话用户useraddpasswdpasswd文件详解 chageusermoduserdelshadow 文件格式切换用户 用户组groupaddgroup文件格式groupmodgroupdel登陆远程机器 磁盘RAIDraid0&#xff08;安装系统&#xff09;raid1&#xff08;存放数据&#xff09;raid 5&#xff0…

盘点54个Python实用工具源码Python爱好者不容错过

盘点54个Python实用工具源码Python爱好者不容错过 学习知识费力气&#xff0c;收集整理更不易。 知识付费甚欢喜&#xff0c;为咱码农谋福利。 链接&#xff1a;https://pan.baidu.com/s/1OXyEh-Yy3JI90jvn6d6wRw?pwd8888 提取码&#xff1a;8888 项目名称 7z辅助破解工…

基于冠状病毒群体免疫算法优化概率神经网络PNN的分类预测 - 附代码

基于冠状病毒群体免疫算法优化概率神经网络PNN的分类预测 - 附代码 文章目录 基于冠状病毒群体免疫算法优化概率神经网络PNN的分类预测 - 附代码1.PNN网络概述2.变压器故障诊街系统相关背景2.1 模型建立 3.基于冠状病毒群体免疫优化的PNN网络5.测试结果6.参考文献7.Matlab代码 …

漂亮的pyqt6皮肤 PyOneDark_Qt_Widgets_Modern_GUIPublic

大家先看看界面图&#xff0c;真的很漂亮&#xff1a; github地址&#xff1a;GitHub - Wanderson-Magalhaes/PyOneDark_Qt_Widgets_Modern_GUI 作者还录了教程&#xff1a; TUTORIALS: Tutorial 01: https://youtu.be/QQGlTGYCMg0 Tutorial 02: https://youtu.be/LwKre2proDk…

Halcon (4):如何开始自学

文章目录 文章专栏前言Halcon文档Halcon基础案例文档英语阅读建议 结论 文章专栏 Halcon开发 前言 在我完成上一篇代码&#xff0c;halcon基础窗口事件写完了之后&#xff0c;我已经基本掌握了如何写一个简单的halcon程序。后面我学习新的知识的时候感觉遇到了瓶颈。因为网上没…

【C++】:模板进阶

朋友们、伙计们&#xff0c;我们又见面了&#xff0c;本期来给大家解读一下有关C模板进阶的知识点&#xff0c;如果看完之后对你有一定的启发&#xff0c;那么请留下你的三连&#xff0c;祝大家心想事成&#xff01; C 语 言 专 栏&#xff1a;C语言&#xff1a;从入门到精通 数…