Largest Rectangle in a Histogram

http://poj.org/problem?id=2559

题意:给出若干宽度相同的矩形的高度(条形统计图),求最大子矩形面积

解题思路

单调栈的经典题(嫌弃字多可以先看后面的示例再来看这个思路

显然,最终的子矩形高度一定和某一个矩形相等(反证)。因此一个暴力的做法就是枚举每一个矩形,然后往两边扩散。在它左侧找到第一个高度比它小的,右侧也一样。则可以求出最大可扩散宽度,乘上高度即可以更新答案。复杂度O(n2)

如果说要优化刚才的算法,也就可以优化寻找最大可扩散宽度的速度

让每一个矩形依次入栈,保存两个关键字:矩形高度,其最大左扩散宽度。保证栈内的矩形高度单调递增

我们可以得到结论:目前栈内的一个矩形uu,在原图中从uu开始一直到栈顶所在的矩形,高度一定都比uu高。这就是为什么我们不需要统计栈内矩形的最大右扩散宽度,因为最大右扩散宽度就是栈顶

再回忆一下最大左扩散宽度的意义,是在它左侧的高度大于它的矩形们。这让我们又得出一个结论:目前栈内的一个矩形uu,如果它的最大左扩散宽度大于11,则这些它所能扩散到的矩形一定都不在栈中。这也很容易发现,因为栈是单调递增的。或者,我们可以得到一个更形象的结论:栈内连续的两个矩形u,vu,v,如果在原图中他们之间有矩形,那么这些矩形一定都高于u,vu,v

因此刚才我们所说的最大左扩散宽度,其实等同于在原图中,它到栈中上一个矩形之间相隔了多少矩形

当一个新的矩形进来的时候,它会弹走若干个矩形。而栈内一个矩形实际上代表着原图中一段矩形。因此可以说是弹走了几段矩形。但是这些被弹走的矩形只不过出栈,在原图中并不会消失。因此他们所代表的的宽度不应当消失,所以我们将他们累积在新进来的这个矩形上。这也非常符合事实——这个新的矩形之所以能弹走这若干个矩形是因为自己比他们矮,因此都可以扩散到。换句话说,被弹走的这一系列矩形最多只能向右扩散到这个新矩形,因此留着它们就没有意义了

而对于任何一个要出栈的矩形,我们需要统计由它的高度所能扩散出去的最大子矩形面积。由于它的最大左扩散宽度已知,唯一需要知道的就是它的最大右扩散宽度。那么由于它在栈里,它的最大右扩散宽度也就是从它一直到最早先的栈顶之间的宽度。因此我们只需要在弹栈的过程中一路累积每个出栈矩形的最大左扩散宽度,加起来就是这一段宽度了。

另外,如果处理完了最后一个矩形以后栈依然有剩余,则应当弹完并更新答案

保证了每个矩形入栈以及出栈恰好一次,在正确性显然的条件下,复杂度O(n)


这里给出示例,以帮助理解:

例子就用题目中的[2,1,5,6,2,3]

首先,如果栈是空的,那么索引i入栈。那么第一个i=0就进去吧。注意栈内保存的是索引,不是高度。然后i++。

然后继续,当i=1的时候,发现h[i]小于了栈内的元素,于是出栈。(由此可以想到,哦,看来stack里面只存放单调递增的索引

这时候stack为空,所以面积的计算是h[t] * i.t是刚刚弹出的stack顶元素。也就是蓝色部分的面积。

继续。这时候stack为空了,继续入栈。注意到只要是连续递增的序列,我们都要keep pushing,直到我们遇到了i=4,h[i]=2小于了栈顶的元素。

这时候开始计算矩形面积。首先弹出栈顶元素,t=3。即下图绿色部分。

接下来注意到栈顶的(索引指向的)元素还是大于当前i指向的元素,于是出栈,并继续计算面积,桃红色部分。

最后,栈顶的(索引指向的)元素大于了当前i指向的元素,循环继续,入栈并推动i前进。直到我们再次遇到下降的元素,也就是我们最后人为添加的dummy元素0.

同理,我们计算栈内的面积。由于当前i是最小元素,所以所有的栈内元素都要被弹出并参与面积计算。

注意我们在计算面积的时候已经更新过了maxArea。

Code

这里如果一下子理解不过来的话,wid可以分两步求,wid = l + r

r = j-t; //右边能扩散的最大宽度
l = s.empty()?t:(t-s.top()-); //左边能扩大的最大宽度
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <stack>
using namespace std; typedef long long ll;
ll h[]; stack<int> s; int main()
{
int n;
while (scanf("%d", &n)&&n) {
memset(h, , sizeof(h));
while(!s.empty()) s.pop();
for (int i = ; i < n; i++)
scanf("%lld", &h[i]); ll j = , res = ;
while (j<=n) {
if (s.empty()||h[s.top()]<=h[j])
s.push(j++);
else {
ll t = s.top(); s.pop();
ll wid = s.empty()?j:(j-s.top()-);
res = max(res, h[t]*wid);
}
}
printf("%lld\n", res);
}
}

这题算法是没什么问题的,但是特别要注意的是,如何不单独写成一个函数的话,由于最后push进去了h值为0的n,这个对之后wid求值时候可能为负数就会出现问题,还有就是要保证每组数据输入完后h[n]要为0

 Largest Submatrix of All 1’s

http://poj.org/problem?id=3494

题意:给你一个n*m的矩阵,矩阵的每个位置值为0或者1,问你在这个矩阵中全部由1组成的最大的矩形面积为多少。
思路:很巧妙的方法。把图拆成一行行来做,分别求以每行为底的最大矩形面积

矩阵中的子矩阵等同于序列中的子序列,只不过此题要做一个预处理,将这个矩阵分为以每一行为x轴n个柱状图,对于每一个等于1的点,它的高度都等于上一行同一列的点的高度加一。

初始化后,再对n个柱状图进行如上题POJ2559的处理。
例如:

 
例子
初始化后
 

以第一行为 x 轴
 

以第二行为 x 轴
 
以第三行为 x 轴
Code
#include<stdio.h>
#include<stack>
using namespace std; int h[][];
stack<int> s; int main()
{
int n, m;
while(~scanf("%d%d",&n,&m)) {
for(int i = ; i < n; i++) {
for(int j = ; j < m; j++) {
scanf("%d",&h[i][j]);
if(h[i][j]==&&i>) h[i][j]+=h[i-][j];
}
}
for(int i = ; i < n; i++) h[i][m] = ; int res = ;
for (int i = ; i < n; i++) {
while(!s.empty()) s.pop();
int j = ;
while (j<=m) {
if (s.empty()||h[i][s.top()]<=h[i][j])
s.push(j++);
else {
int t = s.top(); s.pop();
int wid = s.empty()?j:(j-s.top()-);
res = max(res, h[i][t]*wid);
}
}
} printf("%d\n", res);
}
return ;
}


参考自以下博客:

https://www.cnblogs.com/qixingzhi/p/9497208.html

https://www.cnblogs.com/lichen782/p/leetcode_Largest_Rectangle_in_Histogram.html

https://www.cnblogs.com/boring09/p/4231906.html

[POJ2559&POJ3494] Largest Rectangle in a Histogram&Largest Submatrix of All 1’s 「单调栈」的更多相关文章

  1. ☆ [POJ2559] Largest Rectangle in a Histogram 「单调栈」

    类型:单调栈 传送门:>Here< 题意:给出若干宽度相同的矩形的高度(条形统计图),求最大子矩形面积 解题思路 单调栈的经典题 显然,最终的子矩形高度一定和某一个矩形相等(反证).因此一 ...

  2. POJ2559 Largest Rectangle in a Histogram —— 单调栈

    题目链接:http://poj.org/problem?id=2559 Largest Rectangle in a Histogram Time Limit: 1000MS   Memory Lim ...

  3. 【题解】Largest Rectangle in a Histogram [SP1805] [POJ2559]

    [题解]Largest Rectangle in a Histogram [SP1805] [POJ2559] [题目描述] 传送: \(Largest\) \(Rectangle\) \(in\) ...

  4. poj2559 Largest Rectangle in a Histogram(单调栈)

    Description A histogram is a polygon composed of a sequence of rectangles aligned at a common base l ...

  5. POJ2559 Largest Rectangle in a Histogram (单调栈

    Largest Rectangle in a Histogram Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 26012 ...

  6. NYOJ-258/POJ-2559/HDU-1506 Largest Rectangle in a Histogram,最大长方形,dp或者单调队列!

                                         Largest Rectangle in a Histogram 这么经典的题硬是等今天碰到了原题现场懵逼两小时才会去补题.. ...

  7. poj 2559 Largest Rectangle in a Histogram - 单调栈

    Largest Rectangle in a Histogram Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 19782 ...

  8. DP专题训练之HDU 1506 Largest Rectangle in a Histogram

    Description A histogram is a polygon composed of a sequence of rectangles aligned at a common base l ...

  9. Largest Rectangle in a Histogram(DP)

    Largest Rectangle in a Histogram Time Limit : 2000/1000ms (Java/Other)   Memory Limit : 65536/32768K ...

随机推荐

  1. Java基础(四) StringBuffer、StringBuilder原理浅析

    StringBuilder与StringBuffer作用就是用来处理字符串,但String类本身也具备很多方法可以用来处理字符串,那么为什么还要引入这两个类呢? 关于String的讲解请看Java基础 ...

  2. # 构建以及运行Springboot Docker镜像时的变量传递

    Docker可以把我们的运行环境打包,然后我们只要run就可以了.大部分hello world都是这么写的.但都缺少了实际应用环节.以springboot为例,hello world的Dockerfi ...

  3. 简单有趣的hover

    一张图片在鼠标悬浮的时候换另一张图片 平常写张图片用img我们换一种方法用背景,然后就悬浮出 我们在加个边框试试 怎么才能让边框一起换哪? 我们嵌套一下 然后打开就是 是不是一个很简单的hover

  4. spring boot 2.x 系列 —— spring boot 整合 druid+mybatis

    源码Gitub地址:https://github.com/heibaiying/spring-samples-for-all 一.说明 1.1 项目结构 项目查询用的表对应的建表语句放置在resour ...

  5. iOS开发如何避免安全隐患

    现在很多iOS的APP没有做任何的安全防范措施,导致存在很多安全隐患和事故,今天我们来聊聊iOS开发人员平时怎么做才更安全. 一.网络方面 用抓包工具可以抓取手机通信接口的数据.以Charles为例, ...

  6. [python] 安装TensorFlow问题 解决Cannot uninstall 'wrapt'. It is a distutils installed project

    cmd安装 pip install tensorflow 1.遇到了 ERROR: Cannot uninstall 'wrapt'. It is a distutils installed proj ...

  7. 深度残差网络(ResNet)

    引言 对于传统的深度学习网络应用来说,网络越深,所能学到的东西越多.当然收敛速度也就越慢,训练时间越长,然而深度到了一定程度之后就会发现越往深学习率越低的情况,甚至在一些场景下,网络层数越深反而降低了 ...

  8. html、javascript、url特殊字符的转义诠释及使用方法详解

    html.javascript.url特殊字符转义在实际编程中都是有用到的,有的人对特殊字符转义的使用不是很清楚,下面就对html,javascript,url特殊字符的转义做一下说明和归纳. htm ...

  9. nginx反向代理中神奇的斜线

    nginx反向代理中神奇的斜线 在进行nginx反向代理配置的时候,location和proxy_pass中的斜线会造成各种困扰,有时候多一个或少一个斜线,就会造成完全不同的结果,所以特地将locat ...

  10. CRM 总结

    目录 一. CRM客户关系管理系统 1. CRM是什么? 里面都有哪些功能(业务)? 2. 什么是公户?什么是私户?为什么要做这个区分? 3. 请列举出CRM系统中的表 4. 通过ORM操作对数据库的 ...