poj 2559(栈的应用)
参考资料:
[1]:挑战程序设计竞赛
•题意
柱状图是由一些宽度相等的长方形下端对齐后横向排列得到的图形。
现在有由 n 个宽度为1,高度分别为h[1,2,3.......n]的长方形从左到右依次排列组成的柱状图。
问里面包含的长方形的最大面积是多少?
•题解
如果确定了长方形的左端点L和右端点R,那么最大可能的高度就是min{h[i] | L <= i <= R}。
这样我们就得到了一个O(N3)的算法,如果对计算区间最小值进行一些优化,那么可以把复杂度将为O(N^2)。
但即使是这样,仍然无法在规定时间内求出答案。那么我们应该怎么做才能更高效的求解呢?
设面积最大的长方形左端是L,右端是R,高度是H。
易得H[L-1] < H 且 H[R+1] < H ,H=min{h[ i ] | L <= i <= R} 。
证明:
如果H[L-1] >= H ,那么左端点就可以更新为L-1,从而可以得到更大的长方形,与假设矛盾,因此 H[L-1] < H;
同理可得 H[R+1] < H。
我们可以遍历一边,找到每个 i (i=1,2,3,......,n) 的最小的L[ i ]和最大的R[ i ];
这样答案就是 max( h[i]*(R[i]-L[i]+1) ) (i=1,2,3,.........,n)。
关键就是如何在线性时间内求出每个 i 的 L[ i ]和R[ i ]。
由 H[L-1] < H && H[R+1] < H 可得:
L[i]=( i 之前的高度第一个小于 h[i] 对应的下标) + 1;
R[i]=( i 之后的高度第一个小于 h[i] 对应的下标) - 1;
暴力方法当然是对于每个 i 都遍历一边 i 之前的值和 i 之后的值,这当然是会超时的,所以,我们要换个思路。
引入一个新的数据结构栈;
在计算 L[ i ] 时,首先,判断栈顶元素 j 的高度 h[ j ] 是否大于等于 h[ i ];
如果h[ j ] ≥ h[ i ],则不断弹出栈顶元素,直到 h[ j ] < h[ i ] 或栈为空。
若栈为空,则L[ i ] = 1,反之,L[ i ]=j+1,然后将 i 压入栈中。
计算 R[ i ] 时只需反向( i 从n 到 1 )重复上述过程即可。
由于栈的压入和弹出操作都是 O(N),因此整个算法的时间复杂度为 O(N);
•Code
#include<iostream>
#include<cstdio>
#include<stack>
using namespace std;
#define ll long long
const int maxn=1e5+; int n;
ll h[maxn];
int l[maxn];
int r[maxn];
stack<int >sta; void Clear()
{
while(!sta.empty())
sta.pop();
}
ll Solve()
{
Clear();
for(int i=;i <= n;++i)
{
while(!sta.empty() && h[sta.top()] >= h[i])
sta.pop(); l[i]=sta.empty() ? :sta.top()+;
sta.push(i);
} Clear();
for(int i=n;i >= ;--i)
{
while(!sta.empty() && h[sta.top()] >= h[i])
sta.pop(); r[i]=sta.empty() ? n:sta.top()-;
sta.push(i);
} ll ans=;
for(int i=;i <= n;++i)
ans=max(ans,h[i]*(r[i]-l[i]+)); return ans;
}
int main()
{
while(~scanf("%d",&n) && n)
{
for(int i=;i <= n;++i)
scanf("%lld",h+i); printf("%lld\n",Solve());
}
return ;
}
poj 2559(栈的应用)的更多相关文章
- [POJ 2559]Largest Rectangle in a Histogram 题解(单调栈)
[POJ 2559]Largest Rectangle in a Histogram Description A histogram is a polygon composed of a sequen ...
- poj 2559 Largest Rectangle in a Histogram 栈
// poj 2559 Largest Rectangle in a Histogram 栈 // // n个矩形排在一块,不同的高度,让你求最大的矩形的面积(矩形紧挨在一起) // // 这道题用的 ...
- stack(数组模拟) POJ 2559 Largest Rectangle in a Histogram
题目传送门 /* 题意:宽度为1,高度不等,求最大矩形面积 stack(数组模拟):对于每个a[i]有L[i],R[i]坐标位置 表示a[L[i]] < a[i] < a[R[i]] 的极 ...
- POJ 2559 Program C
Submit Status Practice POJ 2559 Description A histogram is a polygon composed of a sequence of recta ...
- poj 2559 Largest Rectangle in a Histogram (单调栈)
http://poj.org/problem?id=2559 Largest Rectangle in a Histogram Time Limit: 1000MS Memory Limit: 6 ...
- POJ 2559 Largest Rectangle in a Histogram(单调栈)
[题目链接] http://poj.org/problem?id=2559 [题目大意] 给出一些宽度为1的长方形下段对其后横向排列得到的图形,现在给你他们的高度, 求里面包含的最大长方形的面积 [题 ...
- 题解 POJ 2559【Largest Rectangle in a Histogram】(单调栈)
题目链接:http://poj.org/problem?id=2559 思路:单调栈 什么是单调栈? 单调栈,顾名思义,就是单调的栈,也就是占中存的东西永远是单调(也就是递增或递减)的 如何实现一个单 ...
- POJ 2559 Largest Rectangle in a Histogram(单调栈) && 单调栈
嗯... 题目链接:http://poj.org/problem?id=2559 一.单调栈: 1.性质: 单调栈是一种特殊的栈,特殊之处在于栈内的元素都保持一个单调性,可能为单调递增,也可能为单调递 ...
- poj 2559 Largest Rectangle in a Histogram - 单调栈
Largest Rectangle in a Histogram Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 19782 ...
随机推荐
- Puppet日常总结
在工作中常常会有这样一种需求:某几个人需要某些测试服务器的root权限.比如,开发部门的张三,李四,王五,赵六需要rsync服务器的root权限.有些同学会说那直接 visudo在里面添加几个人不就行 ...
- c++ 实现哈夫曼树中遇见的问题
为了提高效率求得 叶子 节点中权值最小的两个元素,我们需要使用堆数据结构,它可以以O(logn)的复杂度 取得n个元素中的最小元素.为了绕过堆的实现,我们可以使用标准模板库中相应的标准模板—优先队列. ...
- 训练赛-Move Between Numbers
题意:给你n个数,每个数有20个数字,每两个数字之间如果相等的数字数量为17个(一定是17),就能从一个数字到达另一个数字,给你两个数字编号,求从第一个数字编号到第二个数字编号之间最少需要走几次: 解 ...
- 【转】微信小程序开发之图片等比例缩放 获取屏幕尺寸图片尺寸 自适应
原文[https://blog.csdn.net/qq_31383345/article/details/53127804] 早上在论坛上看到有人写了关于图片等比例缩放的文章,只是判断了图片宽是否大于 ...
- git 出现错误时
Your local changes to the following files would be overwritten by merge: 解决办法 如果希望保留生产服务器上所做的改动,仅仅并入 ...
- BZOJ2142礼物——扩展卢卡斯
题目描述 一年一度的圣诞节快要来到了.每年的圣诞节小E都会收到许多礼物,当然他也会送出许多礼物.不同的人物在小E 心目中的重要性不同,在小E心中分量越重的人,收到的礼物会越多.小E从商店中购买了n件礼 ...
- Server socket
用法都一样 区别: self.request TCP self.requsst 代表具体的链接 UDP self.requst 代表一个小元组(元组里面: 第一个元素 客户端发来的数据 ,第 ...
- Hdoj 1102.Constructing Roads 题解
Problem Description There are N villages, which are numbered from 1 to N, and you should build some ...
- Hdoj 4540.威威猫系列故事——打地鼠 题解
Problem Description 威威猫最近不务正业,每天沉迷于游戏"打地鼠". 每当朋友们劝他别太着迷游戏,应该好好工作的时候,他总是说,我是威威猫,猫打老鼠就是我的工作! ...
- 故障排错-ping dup!
ping DUP! ping一个vc中虚拟机的地址发现如下,出现了DUP! . 解决方式如下: 1.根据mac地址找到虚拟机网卡的端口组 然后编辑绑定和故障切换,切换负责平衡