LeetCode 84 柱状图中最大的矩形

题目描述

柱状图中最大的矩形

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

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

示例 1:

复制代码
输入:heights = [2,1,5,6,2,3]
输出:10
解释:最大的矩形为图中红色区域,面积为 10

示例 2:

复制代码
输入: heights = [2,4]
输出: 4

提示:

  • 1 <= heights.length <=105
  • 0 <= heights[i] <= 104

解法

解法:暴力解法

java代码:

java 复制代码
class Solution {
    public int largestRectangleArea(int[] heights) {
        if (heights.length == 0) {
            return 0;
        }

        int res = 0;
        for (int i = 0; i < heights.length; i++) {
            if (heights[i] == 0) {
                continue;
            }
            // 找左边大于等于当前值的位置,注意先比较前一个位置的值,再++,防止越界
            int l = i;
            while (l > 0 && heights[l - 1] >= heights[i]) {
                l --;
            }

            // 找右边大于等于当前值的位置,注意先比较后一个位置的值,再++,防止越界
            int r = i;
            while (r < heights.length - 1 && heights[r + 1] >= heights[i]) {
                r ++;
            }

            // 计算当前位置高度的最大矩形面积
            int area = (r - l + 1) * heights[i];
            res = Math.max(area, res);
        }

        return res;
    }
}

复杂度

  • 时间复杂度:O(N^2) N是输入数组的长度
  • 空间复杂度:O(1)

解法2:单调栈

java

java 复制代码
class Solution {
    public int largestRectangleArea(int[] heights) {
        int len = heights.length;
        if (len == 0) {
            return 0;
        }
        if (len == 1) {
            return heights[0];
        }

        int res = 0;
        Deque<Integer> stack = new ArrayDeque<>();

        // 第一遍循环,入栈,并把能计算出来的先计算了(就是当前元素小于栈顶元素了,栈顶元素是可以计算的了)
        for (int i = 0; i < len; i++) {
            // 这个 while 很关键,因为有可能不止一个柱形的最大宽度可以被计算出来
            while (!stack.isEmpty() && heights[i] < heights[stack.peekLast()]) {
                int curHeight = heights[stack.pollLast()];
                while (!stack.isEmpty() && heights[stack.peekLast()] == curHeight) {
                    stack.pollLast();
                }

                int curWidth;
                if (stack.isEmpty()) {
                    curWidth = i;
                } else {
                    curWidth = i - stack.peekLast() - 1;
                }
                res = Math.max(res, curHeight * curWidth);
            }
            stack.addLast(i);
        }

        // 第二遍,栈不为空,继续弹栈
        while (!stack.isEmpty()) {
            int curHeight = heights[stack.pollLast()];
            while (!stack.isEmpty() && heights[stack.peekLast()] == curHeight) {
                stack.pollLast();
            }
            int curWidth;
            if (stack.isEmpty()) {
                curWidth = len;
            } else {
                curWidth = len - stack.peekLast() - 1;
            }
            res = Math.max(res, curHeight * curWidth);
        }

        return res;
    }
}
相关推荐
Sun_light10 分钟前
队列:先进先出的线性数据结构及其应用
前端·javascript·算法
codeRichLife30 分钟前
Mybatisplus3.5.6,用String处理数据库列为JSONB字段
java·数据库
来自星星的猫教授36 分钟前
Java 文件注释规范(便于生成项目文档)
java·注释
吕小鸣37 分钟前
Coze、Dify、FastGPT三大AI智能平台架构与能力对比
算法
zhimeng340 分钟前
自己学习原理
java
程序员鱼皮43 分钟前
学 Java 还是 Go 语言?这事儿很简单!
java·后端·计算机·程序员·开发·编程经验·自学编程
Lanqing_07601 小时前
淘宝商品详情图API接口返回参数说明
java·服务器·前端·api·电商
jndingxin1 小时前
c++ 面试题(1)-----深度优先搜索(DFS)实现
c++·算法·深度优先
矮油0_o1 小时前
第一部分 -- ①语法分析的概要
java·编译器·解释器·语法分析
写bug写bug1 小时前
Dubbo中SPI机制的实现原理和优势
java·后端·dubbo