题目:84. 柱状图中最大的矩形

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

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

示例:

输入: [2,1,5,6,2,3]

输出: 10




我们要求的是能勾勒出来的最大矩形面积

先用暴力破解整理一下思路,可以这样来做,循环遍历每根柱子,以每根柱子为高,不断向两边扩散直到遇到高度比自己低的柱子为止,求出最大面积

//核心代码如下
//从左往右遍历每根柱子
for (int i = 0; i < heights.length; i++) {
int w = 1;
int j = i - 1;
//向左扩散,直到遇到比自己高度低的柱子,宽度不断加一
while (j >= 0) {
if (heights[j] < heights[i]) {
break;
}else {
w++;
j--;
}
}
j = i + 1;
//向右遍历,直到遇到比自己低的柱子,宽度继续增加
while (j < heights.length) {
if (heights[j] < heights[i]) {
break;
}else {
w++;
j++;
}
}
//计算出来以这根柱子为高所能勾勒出来的面积,再比较赋值
maxArea = Math.max(maxArea,heights[i] * w);
}

每根柱子都会向左向右扩散,这样就造成了很多的重复,时间复杂度为O(N²),会超时

这时候再想能不能优化一下,使用更少的时间来解决问题

当使用暴力破解的时候,我们从左往右遍历每一根柱子进行求值

  因为柱子是从左边开始遍历过来的,所以遍历到某值时它左边的所有柱子的高度其实已经被遍历过了,所以我们可以用一个有序的栈来存储左边柱子的值,在这里我们使用递增的栈来存储左边元素

  当我们遍历到比栈顶元素高度更低的柱子时(反之,则入栈),说明以当前高度已经无法向右边扩撒了,而因为栈中元素是递增的,所以左边能扩散到的宽度就是到栈中的上一个元素 (先忽略柱子高度相等的情况),那么栈顶元素所能勾勒出来的最大矩形面积就可以求出来了


我们通过代码来进一步理解

class Solution {
public int largestRectangleArea(int[] heights) {
// 使用栈解决
// 注意:栈里面存储的是下标值,因为高度可以由下标值得出
int res = 0;
int len = heights.length;
Deque<Integer> stack = new ArrayDeque<>(); // 从左往右遍历每一根柱子
for (int i = 0; i < len; i++) {
// 当此时栈顶元素对应柱子的高度大于正在遍历的柱子高度时,进入循环
while (!stack.isEmpty() && heights[stack.peekLast()] > heights[i]) {
// 得出栈顶柱子的高度,并且将起弹出栈
int h = heights[stack.pollLast()]; // 特殊情况,如果现在栈顶柱子的高度与刚弹出栈的柱子的高度相等,那么将其弹出栈
while (!stack.isEmpty() && heights[stack.peekLast()] == h) {
stack.removeLast();
} // w 是所求柱子的宽度,如果栈不为空,它就等于 i 的值减去现在栈顶的值再减一
int w = 0;
if (stack.isEmpty()){
w = i;
} else {
w = i - stack.peekLast() - 1;
}
res = Math.max(res, h * w);
}
stack.addLast(i);
} // 此时栈中可能仍有元素,我们需要计算出以每根柱子为高能勾勒出最大的矩形面积,所以需要将所有元素都弹出栈
while (!stack.isEmpty()) {
int h = heights[stack.pollLast()];
int w = 0;
if (stack.isEmpty()) {
w = len;
} else {
w = len - stack.peekLast() - 1;
}
res = Math.max(res, h * w);
}
return res;
}
}

此时本题已经完成,但我们仍然可以继续进行优化---加入哨兵

比如说 [2,1,5,6,2,3] 是我们要求的柱状图,我们可以在数组的两端都插入 0 ,也就是在原先柱状图的基础上,在两端插入分别插入一个高度为0的柱子,变成[0,2,1,5,6,2,3,0],这样 0 就成为了最小的高度,那么遍历元素结束的时候,所有元素都可以弹出栈,就不用再考虑遍历完栈不为空的情况了

代码:

class Solution {
public int largestRectangleArea(int[] heights) {
// 加入哨兵进行优化
int res = 0;
int len = heights.length;
int[] newHeights = new int[len+2];
newHeights[0] = 0;
System.arraycopy(heights,0,newHeights,1,len);
heights = newHeights;
Deque<Integer> stack = new ArrayDeque<>();
stack.addLast(newHeights[0]);
len = len + 2;
for (int i = 1; i < len; i++) {
while (heights[i] < heights[stack.peekLast()]) {
int h = heights[stack.pollLast()];
int w = i - stack.peekLast() - 1;
res = Math.max(res, h * w);
}
stack.addLast(i);
}
return res;
}
}

LeetCode---84. 柱状图中最大的矩形(hard)的更多相关文章

  1. LeetCode 84. 柱状图中最大的矩形(Largest Rectangle in Histogram)

    84. 柱状图中最大的矩形 84. Largest Rectangle in Histogram

  2. Java实现 LeetCode 84 柱状图中最大得矩形

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

  3. LeetCode 84. 柱状图中最大的矩形(Largest Rectangle in Histogram)

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

  4. leetcode 84. 柱状图中最大的矩形 JAVA

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

  5. [LeetCode] 84. 柱状图中最大的矩形

    题目链接 : https://leetcode-cn.com/problems/largest-rectangle-in-histogram/ 题目描述: 给定 n 个非负整数,用来表示柱状图中各个柱 ...

  6. Leetcode84. 柱状图中最大的矩形(单调栈)

    84. 柱状图中最大的矩形 前置 单调栈 做法 连续区间组成的矩形,是看最短的那一块,求出每一块左边第一个小于其高度的位置,右边也同理,此块作为最短限制.需要两次单调栈 单调栈维护递增区间,每次不满足 ...

  7. 【LeetCode】84. Largest Rectangle in Histogram 柱状图中最大的矩形(Python)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 单调栈 日期 题目地址: https://leetc ...

  8. LeetCode 84. Largest Rectangle in Histogram 单调栈应用

    LeetCode 84. Largest Rectangle in Histogram 单调栈应用 leetcode+ 循环数组,求右边第一个大的数字 求一个数组中右边第一个比他大的数(单调栈 Lee ...

  9. LeetCode(84): 柱状图中最大的矩形

    Hard! 题目描述: 给定 n 个非负整数,用来表示柱状图中各个柱子的高度.每个柱子彼此相邻,且宽度为 1 . 求在该柱状图中,能够勾勒出来的矩形的最大面积. 以上是柱状图的示例,其中每个柱子的宽度 ...

随机推荐

  1. JVM笔记 -- JVM经历了什么?

    Sun Classic VM 世界上第一款商用 Java 虚拟机,JDK1.4 已经淘汰. 内部只有解释器,可以自己外挂JIT编译器,但是二者只能使用其一,不能配合工作. hotspot 内置了该虚拟 ...

  2. 【翻译】内部API的价值

    内部api的设计,主要是为了简化软件的开发,简化系统和操作过程.目前绝大多数用例是这样的. 内部api经常被忽略,因为它们是针对内部开发人员的.这种类型的api通常使用于特定公司及其部门的专用数据.尽 ...

  3. ECMAScript 2016(ES7)新特性简介

    简介 自从ES6(ECMAScript 2015)在2015年发布以来,ECMAScript以每年一个版本的速度持续向前发展.到现在已经是ECMAScript 2020了. 每个版本都有一些新的特性, ...

  4. 浅谈Java的反射的原理

    Java的编译过程 谈及反射,不得不先了解一下,java的整个编译过程,整体的java编译过程可以参考 之前的一篇 一个java文件被执行的历程 这里我们只针对 对象这一层级来讨论,一个java文件, ...

  5. ArrayList源码阅读(小白的java进阶)

    ArrayList(线程不安全) ArrayList是一个其容量能够动态增长的动态数组 继承关系 构造方法 是符合collection父接口的规范的 //传0则设置为默认容量 public Array ...

  6. MySQL入门(4)——操作数据表

    MySQL入门(4)--操作数据表 创建数据库 CREATE [TEMPORARY] TABLE [IF NOT EXISTS] 数据库名 [(create_definition,...)] [tab ...

  7. python中zip函数的使用

    zip(*iterables) zip可以将多个可迭代对象组合成一个迭代器对象,通过迭代取值,可以得到n个长度为m的元组.其中n为长度最短可迭代对象的元素个数,m为可迭代对象的个数.并且每个元组的第i ...

  8. Java自定义 sort 排序方法

    Sort用法 •结构 1 package Test; 2 3 import java.util.Arrays; 4 import java.util.Random; 5 import java.uti ...

  9. 移动端调试vConsole

    当我们在进行移动端开发的时候,经常会出现在pc显示正常,在移动端却各种异常的情况.这时候我们在手机上又看不到error log. 所以我们就需要vConsole这样一个移动端开发神器. 那具体要怎么使 ...

  10. 从零搭建一个IdentityServer——会话管理与登出

    在上一篇文章中我们介绍了单页应用是如何使用IdentityServer完成身份验证的,并且在讲到静默登录以及会话监听的时候都提到会话(Session)这一概念,会话指的是用户与系统之间交互过程,反过来 ...