单调栈题目:柱状图中最大的矩形

news/2024/5/19 1:56:16/文章来源:https://blog.csdn.net/stormsunshine/article/details/121174496

文章目录

  • 题目
    • 标题和出处
    • 难度
    • 题目描述
      • 要求
      • 示例
      • 数据范围
  • 解法
    • 思路和算法
    • 代码
    • 复杂度分析

题目

标题和出处

标题:柱状图中最大的矩形

出处:84. 柱状图中最大的矩形

难度

7 级

题目描述

要求

给定整数数组 heights\texttt{heights}heights 表示柱状图中各个柱子的高度,其中每个柱子的宽度为 1\texttt{1}1,返回在该柱状图中的矩形的最大面积。

示例

示例 1:

示例 1

输入:heights=[2,1,5,6,2,3]\texttt{heights = [2,1,5,6,2,3]}heights = [2,1,5,6,2,3]
输出:10\texttt{10}10
解释:上图为一个柱状图,其中每个柱子的宽度为 1\texttt{1}1。最大的矩形为图中红色区域,面积为 10\texttt{10}10

示例 2:

示例 2

输入:heights=[2,4]\texttt{heights = [2,4]}heights = [2,4]
输出:4\texttt{4}4

数据范围

  • 1≤heights.length≤105\texttt{1} \le \texttt{heights.length} \le \texttt{10}^\texttt{5}1heights.length105
  • 0≤heights[i]≤104\texttt{0} \le \texttt{heights[i]} \le \texttt{10}^\texttt{4}0heights[i]104

解法

思路和算法

为了得到柱状图中最大的矩形,对于下标 iii,需要找到最大的下标 jjj 和最小的下标 kkk,满足 j<i<kj < i < kj<i<kheights[j]<heights[i]\textit{heights}[j] < \textit{heights}[i]heights[j]<heights[i]heights[k]<heights[i]\textit{heights}[k] < \textit{heights}[i]heights[k]<heights[i],则存在一个宽度为 k−j−1k - j - 1kj1、高度为 heights[i]\textit{heights}[i]heights[i] 的矩形,计算该矩形的面积并更新矩形的最大面积。

直观的做法是遍历柱状图中的每个柱子,对于每个下标 iii,向两边遍历寻找对应的下标 jjjkkk,得到以 heights[i]\textit{heights}[i]heights[i] 为高的最大矩形宽度并计算矩形的面积。假设数组 heights\textit{heights}heights 的长度是 nnn,该做法的时间复杂度是 O(n2)O(n^2)O(n2),会超出时间限制,因此需要优化。

创建两个长度为 nnn 的数组 left\textit{left}leftright\textit{right}right,对于每个下标 iiileft[i]\textit{left}[i]left[i]right[i]\textit{right}[i]right[i] 分别记录对应的下标 jjjkkk。初始时,left\textit{left}left 的全部元素为 −1-11right\textit{right}right 的全部元素为 nnn。这道题中可以假设 heights[−1]=heights[n]=−∞\textit{heights}[-1] = \textit{heights}[n] = -\inftyheights[1]=heights[n]=。在遍历数组 heights\textit{heights}heights 的过程中填入 left\textit{left}leftright\textit{right}right 的值。

为了得到 left\textit{left}leftright\textit{right}right 的值,可以使用单调栈,单调栈存储数组 heights\textit{heights}heights 的下标,满足从栈底到栈顶的下标对应的元素单调递增。

从左到右遍历数组 heights\textit{heights}heights,当遍历到下标 iii 时,进行如下操作:

  1. 如果栈不为空且栈顶下标对应的元素大于等于 heights[i]\textit{heights}[i]heights[i],则将栈顶下标出栈,并将栈顶下标对应的 right\textit{right}right 值设为 iii,重复该操作直到栈为空或者栈顶下标对应的元素小于 heights[i]\textit{heights}[i]heights[i]

  2. 如果栈不为空,则栈顶下标对应的元素小于 heights[i]\textit{heights}[i]heights[i],因此将 iii 对应的 left\textit{left}left 值设为栈顶下标;

  3. iii 入栈。

上述操作中,下标 iii 入栈的条件是栈为空或者栈顶下标对应的元素小于 heights[i]\textit{heights}[i]heights[i],因此可以保证从栈底到栈顶的下标对应的元素单调递增。

遍历结束之后,对于每个下标 iii,令 j=left[i]j = \textit{left}[i]j=left[i]k=right[i]k = \textit{right}[i]k=right[i],则有 heights[j]<heights[i]\textit{heights}[j] < \textit{heights}[i]heights[j]<heights[i]heights[k]≤heights[i]\textit{heights}[k] \le \textit{heights}[i]heights[k]heights[i]。如果 heights[k]<heights[i]\textit{heights}[k] < \textit{heights}[i]heights[k]<heights[i],则高度为 heights[i]\textit{heights}[i]heights[i] 的最大矩形宽度即为 k−j−1k - j - 1kj1。如果 heights[k]=heights[i]\textit{heights}[k] = \textit{heights}[i]heights[k]=heights[i],则高度为 heights[i]\textit{heights}[i]heights[i] 的最大矩形宽度大于 k−j−1k - j - 1kj1,但是在遍历到下标 kkk 时可以得到高度为 heights[k]\textit{heights}[k]heights[k] 的矩形宽度为 right[k]−left[k]−1\textit{right}[k] - \textit{left}[k] - 1right[k]left[k]1,该宽度大于 k−j−1k - j - 1kj1。由于一定存在一个柱子,该柱子右边没有和该柱子高度相同的柱子,因此在遍历到该柱子时即可得到以该柱子的高度为矩形高度的矩形最大宽度,并计算得到矩形的最大面积。

对于下标 iii,高度为 heights[i]\textit{heights}[i]heights[i] 的最大矩形的宽度为 right[i]−left[i]−1\textit{right}[i] - \textit{left}[i] - 1right[i]left[i]1(根据上述分析,可能存在 k>ik > ik>i 满足 heights[k]=heights[i]\textit{heights}[k] = \textit{heights}[i]heights[k]=heights[i],但是这里仍然认为该宽度是最大矩形的宽度),面积为 (right[i]−left[i]−1)×heights[i](\textit{right}[i] - \textit{left}[i] - 1) \times \textit{heights}[i](right[i]left[i]1)×heights[i]。遍历 0≤i<n0 \le i < n0i<n,即可得到柱状图中的矩形的最大面积。

考虑一个例子:heights=[2,1,4,4]\textit{heights} = [2, 1, 4, 4]heights=[2,1,4,4]。对应的 left=[−1,−1,1,1]\textit{left} = [-1, -1, 1, 1]left=[1,1,1,1]right=[1,4,3,4]\textit{right} = [1, 4, 3, 4]right=[1,4,3,4],矩形的最大面积是 888

对于下标 000,最大矩形的宽度为 1−(−1)−1=11 - (-1) - 1 = 11(1)1=1,高度为 222,面积为 222

对于下标 111,最大矩形的宽度为 4−(−1)−1=44 - (-1) - 1 = 44(1)1=4,高度为 111,面积为 444

对于下标 222,最大矩形的宽度为 3−1−1=13 - 1 - 1 = 1311=1,高度为 444,面积为 444。注意 heights[2]=heights[3]\textit{heights}[2] = \textit{heights}[3]heights[2]=heights[3],因此最大矩形的宽度应该为 222,但是在下标 222 处计算的最大矩形的宽度为 111,在下标 333 处将会计算到最大矩形的宽度为 222

对于下标 333,最大矩形的宽度为 4−1−1=24 - 1 - 1 = 2411=2,高度为 444,面积为 888

代码

class Solution {public int largestRectangleArea(int[] heights) {int length = heights.length;int[] left = new int[length];int[] right = new int[length];Arrays.fill(left, -1);Arrays.fill(right, length);Deque<Integer> stack = new ArrayDeque<Integer>();for (int i = 0; i < length; i++) {int height = heights[i];while (!stack.isEmpty() && heights[stack.peek()] >= height) {right[stack.pop()] = i;}if (!stack.isEmpty()) {left[i] = stack.peek();}stack.push(i);}int area = 0;for (int i = 0; i < length; i++) {area = Math.max(area, (right[i] - left[i] - 1) * heights[i]);}return area;}
}

复杂度分析

  • 时间复杂度:O(n)O(n)O(n),其中 nnn 是数组 heights\textit{heights}heights 的长度。需要遍历数组 heights\textit{heights}heights 一次,每个下标最多入栈和出栈各一次。

  • 空间复杂度:O(n)O(n)O(n),其中 nnn 是数组 heights\textit{heights}heights 的长度。空间复杂度主要取决于两个数组 left\textit{left}leftright\textit{right}right 以及栈空间,两个数组的长度都是 nnn,栈内元素个数不会超过 nnn

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

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

相关文章

正弦信号发生器的设计

目 录 1 引言 1 2 总体结构设计 2 2.1 单片机概述 2 2.1.1 单片机的发展 2 2.1.2 单片机的用途 3 2.2 系统设计的功能 3 2.3 波形发生和输出频率的方法 4 2.3.1 波形发生的方法 4 2.3.2 输出频率的方法 4 3 系统硬件设计 5 3.1 硬件电路芯片的选择 5 3.1.1 CPU芯片 AT89C51 5 3…

MyBatis中的复杂映射

上一章中实现的MyBatis对象映射较为简单&#xff0c;对象中的属性和数据库中的表字段是一一对应的&#xff08;无论数量和名称都完全一样&#xff09;&#xff0c;如果对象中的属性名和表中的字段名不一致怎么办&#xff1f;又或者Java对象中存在复杂类型属性&#xff08;即类似…

百分点数据科学产教融合计划继续扩大招募

从全球发展来看&#xff0c;数字经济已经成为重组全球要素资源、重塑全球经济结构、改变全球竞争格局的关键力量&#xff0c;是全球共同的发展战略。作为新经济中的数据科学&#xff0c;伴随社会各领域对数字人才需求的与日俱增&#xff0c;也成为了这一波科技革命中的人才竞争…

中国新出海故事:人、疫情与纽带

【潮汐商业评论/原创】 《枪炮、病菌与钢铁》中对人类社会发展的洞察与新千禧年发生了奇妙的呼应&#xff0c;战争、疫情成为了这十年的历史注脚&#xff0c;但时代的车轮总是滚滚向前的&#xff0c;新的世界版图里&#xff0c;全球化、数字化的浪潮构成了新世界跳动的脉搏&am…

第2章 ROS 通信机制 4 —— 常用命令

文章目录1 应用场景2 rosnode 功能包 plumbing_pub_sub编译执行3 rostopic 功能包 plumbing_pub_sub编译执行4 rosservice (服务通信) 功能包 plumbing_server_client编译执行5 rosmsg (话题通信) 功能包 plumbing_pub_sub编译执行6 rossrv (服务通信) 功能包 plumbing_server_…

常见网络知识面试题总结

&#x1f353;个人主页&#xff1a;个人主页 &#x1f352;系列专栏&#xff1a;C/C基础与进阶 &#x1f4ac;推荐一款模拟面试、刷题神器&#xff0c;从基础到大厂面试题&#x1f449;点击跳转刷题网站进行注册学习 目录 1、OSI七层模型与TCPIP四层模型是什么&#xff1f; 2…

核爆!字节跳动算法大佬手写1000页数据算法笔记:Github已标星79k

数据结构是什么 数据结构是计算机存储、组织数据的方式。数据结构是指相互之间存在一种或多种特定关系的数据元素的集合。通常情况下&#xff0c;精心选择的数据结构可以带来更高的运行或者存储效率。数据结构往往同高效的检索算法和索引技术有关。 算法是什么 算法是对解题方案…

PC 端网页特效

一、元素偏移量 offset 系列 (一)offset 概述 1、offset 翻译过来就是偏移量,我们使用 offset 系列相关属性可以动态的得到该元素的位置(偏移)、大小等。 (1)获得元素距离带有定位父元素的位置; (2)获得元素自身的大小(宽度高度); (3)注意:返回的数值都不带单位…

使用polkadot.js在substrate frontier上安装ERC20token合约

使用polkadot.js在substrate frontier上安装ERC20token合约参考资料Substrate Frontier Node Template github.com/substrate-developer-hub/frontier-node-template frontier-node-template/examples/contract-erc20/truffle/contracts/MyToken.json安装ERC20 token合约 sourc…

SwiftUI AR教程之应用程序中使用 RealityKit 生成 3D 文本(教程含完整源码)

项目文件 基本设置 我们将从 Xcode 上的“增强现实应用程序”模板开始: 我使用 SwiftUI 作为界面,使用 Swift 作为语言,使用 RealityKit 作为内容技术: 您现在应该拥有基本的增强现实应用程序模板代码。 使用 RealityKit 生成 3D 文本网格 我们将向我们的项目添加一个函…

快速入门SSMS的使用

快速入门SSMS的使用1.Install SSMS2.Demo2.1 Export DDL2.2 XXXXX3.XXXX4.Waken1.Install SSMS 官方地址 SSMS Website: https://learn.microsoft.com/zh-tw/docs/. 也可以直接下载 Download Website: https://aka.ms/ssmsfullsetup.双击安装 要有电脑的管理员权限 快速启动…

springboot基于JAVA游戏周边商城设计与实现毕业设计源码261622

Springboot游戏周边商城的开发 摘 要 现今人们的生活方式逐渐丰富&#xff0c;电脑和网络已经融入了人们生活中的滴滴点点&#xff0c;无时不刻的影响着我们的日常生活&#xff0c;网络游戏已经进入到了大多数人的生活之中。在游戏的世界中人们会得到很多游戏商品&#xff0c;然…

死锁检测组件原理及代码实现

一、引言 所谓死锁&#xff0c;是指多个线程或进程各自持有某些资源&#xff0c;同时又等待着别的线程或进程释放它们现在所保持的资源&#xff0c;否则就不能向前推进。如下图&#xff1a;线程各自占有一把锁&#xff0c;还需要申请别的线程当前持有的锁&#xff0c;形成锁资…

Cisco简单配置(十四)—第一跳冗余协议—HSRP

为什么使用第一跳冗余 默认网关限制 如果路由器或路由器接口&#xff08;作为默认网关&#xff09;发生故障&#xff0c;配置该默认网关的主机将与外部网络隔离。在交换网络中&#xff0c;每个客户端仅收到一个默认网关。即使存在第二个路由可以从本地网段传输数据包&#xf…

微信小程序 java高校新生报到宿舍安排管理系统python php

将小程序权限按管理员和用户这两类涉及用户划分。 (a) 管理员&#xff1b;管理员使用本程序涉到的功能主要有&#xff1a;个人中心、宿舍管理、学生管理、宿舍安排管理、缴费信息管理、程序管理等功能 (b)用户进入程序前台可以实现首页、互助沟通、我的等功能 uni-app框架&…

基于java校园志愿者管理系统(java毕业设计)

基于java校园志愿者管理系统 校园志愿者系统是基于java编程语言&#xff0c;mysql数据库&#xff0c;springboot框架&#xff0c;idea开发工具进行开发&#xff0c;本系统主要分为志愿者和管理员两个角色&#xff0c;其中志愿者的主要功能是查看系统公告&#xff0c;活动信息&…

红队工具合集,安全er值得拥有

背景 圈内很多师傅一直在做红队安全工具箱,用于在hvv、渗透等工作中提升工作效率。依照ATT&CK威胁图谱的指导,我们很容易整理出常用的红队工具合集,在这里为大家展示。 工具介绍 信息搜集 信息搜集一直是渗透测试工作开展的重中之重,找到无人关注的老旧应用,先对…

leetcode 617. Merge Two Binary Trees 合并二叉树(简单)

直接用递归调用给定函数,先判断如果root1为空返回root2,如果root2为空返回root1,都存在的情况下建立新节点node,然后对root1和root2的左子节点调用递归并赋给node的左子节点,再对root1和root2的右子节点调用递归并赋给node的右子节点,返回node即可。一、题目大意 给你两棵…

虚拟机安装

ubuntu 虚拟机安装配置&#xff0c;以 18.04 为例 一、安装步骤 > 安装 vmware wmware 下载地址 &#xff1a; download 点击进入下载界面 点击并下载 windows 平台下的安装包 安装时直接一键下一步即可&#xff0c;也可根据自己需求勾选&#xff0c;最后的注册码可以自行…

吃个晚饭的时间,看明白三相交流感应电机驱动原理

&#x1f495;三相交流感应单机驱动方式 物理开关&#xff1a;&#xff08;接触器开关、正反向控制&#xff0c;星三角启动&#xff09; 变频驱动&#xff1a;&#xff08;软启动、变频器调速、一般无星三角启动&#xff09; &#x1f495;一、物理开关驱动 &#x1f91e;该电…