Largest Rectangle in a Histogram
链接:https://ac.nowcoder.com/acm/contest/1005/D
题目描述
(中午版)
直方图是由在公共基线处对齐的一系列矩形组成的多边形。
矩形具有相等的宽度,但可以具有不同的高度。
例如,图例左侧显示了由高度为2,1,4,5,1,3,3的矩形组成的直方图,矩形的宽度都为1:
通常,直方图用于表示离散分布,例如,文本中字符的频率。
现在,请你计算在公共基线处对齐的直方图中最大矩形的面积。
图例右图显示了所描绘直方图的最大对齐矩形。
输入格式
输入包含几个测试用例。
每个测试用例占据一行,用以描述一个直方图,并以整数 n 开始,表示组成直方图的矩形数目。1≤n≤100000。
然后跟随n个整数h1,…,hn。0≤hi≤1000000000。
这些数字以从左到右的顺序表示直方图的各个矩形的高度。
每个矩形的宽度为1。
同行数字用空格隔开。
当输入用例为n=0时,结束输入,且该用例不用考虑。
输出格式
对于每一个测试用例,输出一个整数,代表指定直方图中最大矩形的区域面积。
每个数据占一行。
请注意,此矩形必须在公共基线处对齐。
输入样例:
7 2 1 4 5 1 3 3
4 1000 1000 1000 1000
0
输出样例:
8
4000
题解
- 主要是单调栈的运用
对于本题而言,如果我们确定了矩形的高度,那么他的长度就确定了
因为矩阵要最大,所以可以根据矩阵高度的单调性求出每个高度对应的 l,r 即可求出长度r - l - 1,然后面积 = 高度 * 长度即可。
最后遍历一遍每个高度对应的面积取最大值即可。
AC代码
#include<stdio.h>
#include<string.h>
#include<stack>
using namespace std;
typedef long long ll;
const int N = 1e5 + 5;
ll l[N], r[N];
ll a[N];
int n;
stack<ll> s;
int main(){
while(scanf("%d",&n) && n){
while(!s.empty()) s.pop();
for(int i=1; i<=n; ++i){
scanf("%lld",&a[i]);
l[i] = r[i] = i;
}
a[0] = a[n+1] = -1;
for(int i=1; i<=n; ++i){
while(!s.empty() && a[i] <= a[s.top()])
s.pop();
if(s.empty()) l[i] = 0;
else l[i] = s.top();
s.push(i);
}
while(!s.empty()) s.pop();
for(int i=n; i>=1; --i){
while(!s.empty() && a[i] <= a[s.top()])
s.pop();
if(s.empty()) r[i] = n+1;
else r[i] = s.top();
s.push(i);
}
ll sum, max = -1;
for(int i=1; i<=n; ++i){
sum = (r[i] - l[i] - 1) * a[i];
if(sum > max) max = sum;
}
printf("%lld\n",max);
}
return 0;
}