DAY60 84.柱状图中最大的矩形
84.柱状图中最大的矩形
题目要求:给定 n 个非负整数,用来表示柱状图中各个柱子的高度。每个柱子彼此相邻,且宽度为 1 。
求在该柱状图中,能够勾勒出来的矩形的最大面积。

思路
单调栈
本地单调栈的解法和接雨水的题目是遥相呼应的。
为什么这么说呢,42. 接雨水 (opens new window)是找每个柱子左右两边第一个大于该柱子高度的柱子,而本题是找每个柱子左右两边第一个小于该柱子的柱子。
这里就涉及到了单调栈很重要的性质,就是单调栈里的顺序,是从小到大还是从大到小。
在题解42. 接雨水 (opens new window)中我讲解了接雨水的单调栈从栈头(元素从栈头弹出)到栈底的顺序应该是从小到大的顺序。
那么因为本题是要找每个柱子左右两边第一个小于该柱子的柱子,所以从栈头(元素从栈头弹出)到栈底的顺序应该是从大到小的顺序!
我来举一个例子,如图:

只有栈里从大到小的顺序,才能保证栈顶元素找到左右两边第一个小于栈顶元素的柱子。
所以本题单调栈的顺序正好与接雨水反过来。
此时大家应该可以发现其实就是栈顶和栈顶的下一个元素以及要入栈的三个元素组成了我们要求最大面积的高度和宽度
理解这一点,对单调栈就掌握的比较到位了。
除了栈内元素顺序和接雨水不同,剩下的逻辑就都差不多了,在题解42. 接雨水 (opens new window)我已经对单调栈的各个方面做了详细讲解,这里就不赘述了。
主要就是分析清楚如下三种情况:
- 情况一:当前遍历的元素heights[i]大于栈顶元素heights[st.top()]的情况
- 情况二:当前遍历的元素heights[i]等于栈顶元素heights[st.top()]的情况
- 情况三:当前遍历的元素heights[i]小于栈顶元素heights[st.top()]的情况
每一次入栈新元素时,我是一直向左边比较比我小的柱子,计算面积并且更新result的。这个思路好神奇我还是没有太想明白中间在while中一直在pop是怎么继续比较的。我个人认为一直在计算的是以每一个i为结尾(right)能够组成的最大矩形长度,这样理解就对了。因为如果当前的i对应的值是2,之前有5,6。即便2之后在出现6,由于2存在的原因也无法组成以5、6为高度的矩形了,所以2能够pop掉2之前所有比2大的st.top()。
class Solution {
public:int largestRectangleArea(vector<int>& heights) {int result = 0;stack<int> st;heights.insert(heights.begin(), 0);heights.push_back(0);st.push(0);for (int i = 1; i < heights.size(); ++i) {if (heights[i] > heights[st.top()]) {st.push(i);} else if (heights[i] == heights[st.top()]) {st.pop();st.push(i);} else if (heights[i] < heights[st.top()]) {while (!st.empty() && heights[i] < heights[st.top()]) {int mid = st.top();st.pop();if (!st.empty()) {int left = st.top();int right = i;int w = right - left - 1;int h = heights[mid];result = max(result, w * h);cout << mid << ' ' << left << ' ' << right << ' ' << w << ' ' << h << ' ' << result << endl;}}st.push(i);}}return result;}
};
细心的录友会发现,我在 height数组上后,都加了一个元素0, 为什么这么做呢?
首先来说末尾为什么要加元素0?
如果数组本身就是升序的,例如[2,4,6,8],那么入栈之后 都是单调递减,一直都没有走 情况三 计算结果的哪一步,所以最后输出的就是0了。 如图:

那么结尾加一个0,就会让栈里的所有元素,走到情况三的逻辑。
开头为什么要加元素0?
如果数组本身是降序的,例如 [8,6,4,2],在 8 入栈后,6 开始与8 进行比较,此时我们得到 mid(8),rigt(6),但是得不到 left。
(mid、left,right 都是对应版本一里的逻辑)
因为 将 8 弹出之后,栈里没有元素了,那么为了避免空栈取值,直接跳过了计算结果的逻辑。
之后又将6 加入栈(此时8已经弹出了),然后 就是 4 与 栈口元素 8 进行比较,周而复始,那么计算的最后结果resutl就是0。 如图所示:

所以我们需要在 height数组前后各加一个元素0。
结束啦,今天就是算法训练营的Day60!
相关文章:
DAY60 84.柱状图中最大的矩形
84.柱状图中最大的矩形 题目要求:给定 n 个非负整数,用来表示柱状图中各个柱子的高度。每个柱子彼此相邻,且宽度为 1 。 求在该柱状图中,能够勾勒出来的矩形的最大面积。 思路 单调栈 本地单调栈的解法和接雨水的题目是遥相呼…...
你知道Linux操作系统的前世今生吗?Linux系统又该如何搭建呢?
文章目录 前言1. Linux 是什么1.1 Unix & Linux 发展历程图1.2 Linux 的发展1.3 Linux 的发行版 2. Linux 环境搭建2.1 环境搭建方式2.2 使用云服务器 3. 使用终端软件连接到 Linux3.1 什么是终端软件3.2 下载安装 XShell3.3 使用 XShell 登陆主机 总结 前言 可能很多人都…...
674. 最长连续递增序列
给定一个未经排序的整数数组,找到最长且 连续递增的子序列,并返回该序列的长度。 连续递增的子序列 可以由两个下标 l 和 r(l < r)确定,如果对于每个 l < i < r,都有 nums[i] < nums[i 1] &a…...
DS5上ARM编译器样例工程改为GCC编译
想问一下,DS5上ARM编译器通过的样例工程,换成aarch64-none-elf-gcc工具链,是不是需要把startup.S改成gcc支持的格式呀?怎么改呢,求助大神们指点一下!谢谢!...
关于Unity Time.deltaTime的理解和使用
Unity中的Time.deltaTime是一个表示上一帧到当前帧所用时间的浮点数。 它可以让Unity应用程序能够以平滑的方式在不同的帧率下运行。 要深刻理解Time.deltaTime,首先得了解Unity引擎得工作原理。 Unity引擎以每秒帧数(FPS)的形式运行。 比…...
Vue3 配置全局 scss 变量
variables.scss $color: #0c8ce9;vite.config.ts // 全局css变量css: {preprocessorOptions: {scss: {additionalData: import "/styles/variables.scss";,},},},.vue 文件使用...
45.120.101.X 如何找出网站建设中弱点和漏洞
漏洞扫描服务(Vulnerability Scan Service)集Web漏洞扫描、操作系统漏洞扫描、资产内容合规检测、配置基线扫描、弱密码检测五大核心功能,自动发现网站或服务器在网络中的安全风险,为云上业务提供多维度的安全检测服务,…...
linux 下打印堆栈信息 jstack pstack gstack 有啥区别?分别的使用场景是啥?
jstack、pstack和gstack是在Linux系统下用于打印堆栈信息的工具,它们的使用场景和功能略有不同。 jstack:jstack是Java虚拟机自带的工具,用于打印Java进程的堆栈信息。它可以显示Java线程的状态、锁信息、线程堆栈等。jstack主要用于诊断Java…...
Vue 3实战:打造交互丰富的任务管理应用
Vue 3实战:打造交互丰富的任务管理应用 前言搭建Vue 3项目步骤 1: 安装Vue CLI 3步骤 2: 创建Vue 3项目步骤 3: 进入项目目录步骤 4: 启动项目步骤 5: 查看项目结构 组件设计与复用1. **组件的职责单一化:**2. **Props传递:**3. **插槽(Slots)…...
python之列表
列表常用操作 l [1,2,3,4,5]# 列表之切片 l1 l[:3] print(l1) # [1, 2, 3],结果为下标0到2 l2 l[3:] print(l2) # [4, 5] ,从下标3开始直到结束 l3 l[1:-1] print(l3) # [2, 3, 4] , 去头去尾...
想要保护服务器的安全,使用哪个软件比较好?
随着互联网的发展普及,网络安全问题也越发凸显,相信不少使用服务器的用户,有遇到过或是听过服务器被入侵导致数据丢失或是被植入病毒木马程序被用来挖矿的情况。那么面对这类情况,我们该如何做好安全工作来保障我们服务器的使用安…...
gitlab图形化界面使用
gitlab使用 创建用户 上面是创建用户基本操作 修改密码 创建组 给组添加用户 创建项目 选择空白项目 退出root用户,切换其他用户 在服务器上创建ssh密钥 使用ssh-ketgen 命令 新服务器上创建的 [rootgitlab ~]# ssh-keygen Generating public/private rsa key …...
Vue使用基本教程(基本介绍及对比,初步使用,构建项目,编辑器等)
一、Vue及与其他前端框架的异同。 Vue.js(通常简称为Vue)是一个用于构建用户界面的渐进式JavaScript框架。它专注于视图层,采用简单的API设计,使得开发者能够更轻松地构建交互式的单页面应用(SPA)和用户界…...
基恩士软件的基本操作(四,快速编辑plc技巧)
目录 单元软原件注释快速添加 双击单元配置,进入单元编辑器 KV一键添加注释 双击软元件注释 进入软元件编辑界面 ,对弹出的列表中软元件打勾点击登录 元件注释就自动添加了 注释收索,快速编辑软元件 自定义注释收索 空软元件快速查找 …...
通达信的ebk文件
我们在通达信软件中 调出 “自定义板块设置” 这个菜单,点击“导出”,会提示你存储 “自选股.EBK”,其实就是对自定义板块里的目录进行备份的一种方式, 当我们打开 这个文件,你会发现其实就是存储了 股票代码ÿ…...
城市易涝点怎么安装万宾科技内涝积水监测仪?
城市内涝是多个城市广泛存在的问题,经常给城市的居民和基础设施带来一些安全威胁。暴雨引发的道路积水和交通中断、财产损失,甚至公共安全威胁都是城市管理者需要提前预防的问题。为了解决这些问题,内涝积水监测仪的应用是一大重要的举措&…...
css取消移动端长按元素背景色
在开发微信小程序的时候,发现有的元素长按之后,出现了讨厌人的背景色,这就很奇怪,就想把它去掉,所以这里教一下方法: 在所在元素添加css样式: // 取消长按的背景色-webkit-tap-highlight-color:…...
inBuilder低代码平台新特性推荐-第九期
各位知乎的友友们,大家好~ 今天来给大家带来的是inBuilder低代码平台特性推荐系列第九期——子表弹出新增! 01 概述 子表弹出新增,是低代码平台提供的一种前端输入组件,在子表字段较多的场景中,有时为了方便…...
C语言——递归实现汉诺塔游戏
归纳编程学习的感悟, 记录奋斗路上的点滴, 希望能帮到一样刻苦的你! 如有不足欢迎指正! 共同学习交流! 🌎欢迎各位→点赞 👍 收藏⭐ 留言📝 比别人多一点努力,你…...
使用MONAI轻松加载医学公开数据集,包括医学分割十项全能挑战数据集和MedMNIST分类数据集
在深度学习中,使用公开数据集具有以下优点: 提供了一个标准化的基准来比较不同算法或模型的性能,因为这些公共数据集被广泛使用,许多研究人员都使用它们来评估他们的方法。可以节省大量的时间和金钱,因为这些数据集已…...
速看|营销智脑 V6 本周上线,四大维度焕新,解锁全域营销新玩法
在 AI 技术飞速迭代的当下,人人都在谈AI商业化,却很少有人真正看透其底层逻辑。从通用大模型横空出世,到各行各业落地AI应用,看似纷繁复杂的技术变革、商业转型,归根结底只在做一件事:把人类漫长积累的认知…...
(初阶) 从零开始:Tushare环境配置与基础数据获取
去年接触到量化投资这个概念时,我面对的第一个问题不是策略怎么写、回测怎么做,而是——数据从哪来?市面上主流的金融数据终端动辄上万一年,对个人量化爱好者来说实在吃不消。 幸运的是,我遇到了Tushare。这是一个完全…...
AS5600磁编码器避坑指南:从I2C通信失败到角度跳变的5个常见问题及解决方法
AS5600磁编码器实战避坑手册:5个高频故障的工程级解决方案 磁编码器在电机控制、机器人关节定位等场景中扮演着关键角色,而AS5600凭借其高性价比和I2C接口的便利性成为许多工程师的首选。但在实际部署中,从I2C通信失败到角度跳变等问题常常让…...
AI搜索时代内容优化实战:GEO工具包审计与结构化数据生成指南
1. 项目概述:为AI搜索时代优化你的内容工具箱 如果你还在用传统的SEO思维做内容,那可能已经落后了。过去一年,我亲眼见证了流量格局的剧变:来自ChatGPT、Perplexity、Copilot这类AI搜索引擎的访问量,正在以惊人的速度…...
NFC技术破局:从黑客松实战到智能场景应用开发
1. 项目概述:一场被巨头押注的技术狂欢在科技圈里待久了,你会发现一个有趣的现象:风口总在变,今天AI,明天元宇宙,但总有一些东西,它们的热度似乎从未真正消退,反而像陈年老酒&#x…...
STM32CubeMX + FreeRTOS 实战:从零到一,手把手教你为STM32F103C8T6搭建一个带LED、按键和串口打印的多任务系统
STM32CubeMX FreeRTOS 实战:构建智能设备控制台的多任务系统 1. 项目概述与硬件准备 想象一下,你正在开发一个智能家居控制器的原型系统。这个系统需要同时处理多个任务:实时监测用户按键输入、控制LED状态指示、通过串口与上位机通信。这正…...
Allwinner A523处理器解析:跨界SoC的性能与应用
1. Allwinner A523处理器深度解析:一款面向平板与嵌入式设备的全能型SoC Allwinner A523这颗八核Cortex-A55处理器最近在嵌入式圈子里引发了广泛讨论。作为深耕ARM架构开发多年的工程师,我认为这款SoC的定位非常巧妙——它既延续了全志在平板电脑市场的传…...
老司机翻车记:双路E5+PVE7.0直通GTX1060,我踩过的那些坑和最终解法
双路E5平台PVE7.0显卡直通实战:从错误码43到完美驱动的深度排错指南 当你在双路E5服务器上尝试将GTX1060直通给PVE7.0虚拟机时,可能会遇到一系列令人抓狂的问题——黑屏、错误码43、分辨率异常、光标闪烁...这些问题往往让中高级用户也束手无策。本文不是…...
小基站、运营商Wi-Fi与光网络融合:2012年通信基础设施变革的技术驱动力与部署实践
1. 市场繁荣背后的技术驱动力解析2012年,当行业报告显示运营商Wi-Fi和光网络市场正在蓬勃发展时,这不仅仅是一个简单的市场数据,它背后反映的是一场由用户行为改变引发的、深刻的基础设施技术变革。作为一名长期跟踪通信网络部署的从业者&…...
日本半导体产业整合困局:从ASIC到ASSP的转型挑战
1. 日本半导体产业整合的迷思与困局2012年初,一则来自日本经济新闻的报道在半导体业界投下了一颗重磅炸弹。报道称,日本三大电子巨头——瑞萨电子、富士通和松下——正计划将其系统级芯片的设计开发部门合并,成立一家全新的公司。与此同时&am…...
