A题 最大矩形
题目描述
给一个直方图,求直方图中的最大矩形的面积。例如,下面这个图片中直方图的高度从左到右分别是2, 1, 4, 5, 1, 3, 3, 他们的宽都是1,其中最大的矩形是阴影部分。
input:
输入包含多组数据。每组数据用一个整数n来表示直方图中小矩形的个数,你可以假定1 <= n <= 100000. 然后接下来n个整数h1, …, hn, 满足 0 <= hi <= 1000000000. 这些数字表示直方图中从左到右每个小矩形的高度,每个小矩形的宽度为1。 测试数据以0结尾。
output:
对于每组测试数据输出一行一个整数表示答案。
样例:
in:
7 2 1 4 5 1 3 3
4 1000 1000 1000 1000
0
out:
8
4000
思路
利用单调栈的思想,得到每一个高度的左右边第一个小于该高度的索引,索引差乘以高度就是矩形面积,比较每一次的大小,输出最大值。
这里具体说一下如何实现单调栈,我使用STL中的stack结构,利用两· 个数组分别储存左右第一个小于该高度的索引,计算右索引时用到的是非递减单调栈,就是说当栈顶元素小于等于待插入元素是才能插入,否则弹栈继续该判断,直至栈空。栈顶元素弹栈,也就意味着栈顶元素已经遇到了右边第一个小于自身的数,更新R[st.top()]。所有的点都加入后,将栈里的所有元素的R值更新为n,即可以延伸到最后。获取左坐标的方法一样,就点的插入顺序变为逆序,即从第n个开始入栈。
代码
#include<iostream>
#include<stdio.h>
#include<stack>
#include<string.h>
#include<algorithm>
#define Max 100000
#define ll long long
using namespace std;
stack<int>st;
ll a[Max+2];
int L[Max+1],R[Max+1],n;
int main()
{
scanf("%d",&n);
while(n)
{
for(int i=0;i<n;i++)
scanf("%lld",&a[i]);
for(int i=0;i<n;i++)
{
while(!st.empty()&&a[st.top()]>a[i])
{
R[st.top()]=i;
st.pop();
}
st.push(i);
}
while(!st.empty())
{
R[st.top()]=n;
st.pop();
}
for(int i=n-1;i>=0;i--)
{
while(!st.empty()&&a[st.top()]>a[i])
{
L[st.top()]=i;
st.pop();
}
st.push(i);
}
while(!st.empty())
{
L[st