hdu1506 直方图中最大的矩形 单调栈入门

直方图是由在公共基线对齐的矩形序列组成的多边形。矩形具有相同的宽度,但可能具有不同的高度。例如,左侧的数字显示了由高度为2,1,4,5,1,3,3的矩形组成的直方图,单位为1,其中矩形的宽度为1:

通常,直方图用于表示离散分布,例如文本中字符的频率。请注意,矩形的顺序,即它们的高度很重要。计算在公共基线上对齐的直方图中最大矩形的面积。右图显示了所描述的直方图的最大对齐矩形。

输入包含几个测试用例。每个测试用例都描述一个直方图,并以整数n开头,表示它组成的矩形的数量。您可以假设1 <= n <= 100000。然后遵循n个整数h1,...,hn,其中0 <= hi <= 1000000000。这些数字表示直方图的矩形的高度,从左到右订购。每个矩形的宽度是1.零跟在最后一个测试用例的输入之后。对于每个测试用例,在一行中输出指定直方图中最大矩形的面积。请记住,这个矩形必须在公共基线上对齐。

单调栈,就是具有单调性质的栈,用途与单调队列有相似之处。我们先来分析一下这道题。一个最大矩阵的高度,就是矩阵覆盖到的,高度最小的条形的高度。所以我们只需要枚举每个条形作为高度最小的条形,所能形成的最大矩阵,然后取最大值即可。

那么现在,问题就转换成了对于每个条形,以它的高度作为矩阵的高度,矩阵最多能向左和向右延伸到的高度,相当于求这个矩阵,左边有连续多少个矩阵比它高,右边有连续多少个矩阵比它高。所以,长度就是:(右边第一个比它小的条形的位置-左边第一个比它小的条形的位置-1)。如何求第一个比当前数小的左侧或右侧数呢?这就要用到单调栈。

设第一个比当前数(序号为i)小的左侧数为\(left[i]\)。算法的流程是这样的:设现在程序循环到第i个数,如果当前,栈顶的数大于等于第i个数,说明什么?说明序号i以后的数j,它们的\(left[j]\)一定不会是栈顶那个数,因为选了j一定能选i。既然这样,直接把j从栈中弹出,然后继续判断……直至第i个数遇到比它小的栈顶的数。这个数就是\(left[i]\),因为比i小的数只会被更小更优的数替换。然后,将i入栈。这样就把那些无用的,值大于第i个数,却在第i个数左边的数统统用第i个数替换了。这就去除了冗余状态。

从这道题,我们能发现一些单调栈和单调队列的共同点和不同点。共同点:1.它们都保持了单调性。2.它们都通过一个元素必定比其它优,就可以直接删去其他元素的方法,去除了冗余状态。3.由于元素都是一进一出,时间复杂度为O(n)。不同点:单调队列,最优值和插入操作在队首,在弹出操作在队尾。单调栈,最优值,插入操作和弹出操作都在栈顶。

#include <cstdio>
using namespace std; const int maxn=1e5+5;
typedef long long LL;
struct stack{
int t, a[maxn];
void reset(){ t=0; }
void pop(){ if (--t==-1) ++t; }
void add(int x){ a[++t]=x; }
int top(){ return a[t]; }
}s; int n;
int a[maxn], lessleft[maxn], lessright[maxn]; int max(const int &a, const int &b){ return a<b?b:a; }
LL max(const LL &a, const LL &b){ return a<b?b:a; } int main(){
while (scanf("%d", &n), n){
a[0]=a[n+1]=-1;
s.reset(); s.add(0);
for (int i=1; i<=n; ++i){
scanf("%d", &a[i]);
while (a[i]<=a[s.top()]) s.pop();
lessleft[i]=s.top(); s.add(i);
}
s.reset(); s.add(n+1);
for (int i=n; i>=1; --i){
while (a[i]<=a[s.top()]) s.pop();
lessright[i]=s.top(); s.add(i);
}
LL maximum=0;
for (int i=1; i<=n; ++i)
maximum=max(maximum,
LL(lessright[i]-lessleft[i]-1)*a[i]);
printf("%lld\n", maximum);
}
return 0;
}

hdu1506 直方图中最大的矩形 单调栈入门的更多相关文章

  1. [LeetCode] Largest Rectangle in Histogram 直方图中最大的矩形

    Given n non-negative integers representing the histogram's bar height where the width of each bar is ...

  2. [LeetCode] 84. Largest Rectangle in Histogram 直方图中最大的矩形

    Given n non-negative integers representing the histogram's bar height where the width of each bar is ...

  3. POJ 3250 Bad Hair Day【单调栈入门】

    Bad Hair Day Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 24112   Accepted: 8208 Des ...

  4. AcWing:131. 直方图中最大的矩形(贪心 + 单调栈)

    直方图是由在公共基线处对齐的一系列矩形组成的多边形. 矩形具有相等的宽度,但可以具有不同的高度. 例如,图例左侧显示了由高度为2,1,4,5,1,3,3的矩形组成的直方图,矩形的宽度都为1: 通常,直 ...

  5. [leetcode]84. Largest Rectangle in Histogram直方图中的最大矩形

    Given n non-negative integers representing the histogram's bar height where the width of each bar is ...

  6. 51nod 1102 面积最大的矩形 (单调栈)

    链接:https://www.51nod.com/onlineJudge/questionCode.html#!problemId=1102 思路: 首先介绍下单调栈的功能:利用单调栈,可以找到从左/ ...

  7. [51nod1102]面积最大的矩形(单调栈||预处理)

    题意:求序列上某区间最小值乘区间长度的最大值. 解题关键:很早就在<挑战程序设计竞赛>中见过了,单调栈模板题,注意弹栈时如何处理后面的元素. 法一:单调栈 #include<bits ...

  8. Leetcode84. 柱状图中最大的矩形(单调栈)

    84. 柱状图中最大的矩形 前置 单调栈 做法 连续区间组成的矩形,是看最短的那一块,求出每一块左边第一个小于其高度的位置,右边也同理,此块作为最短限制.需要两次单调栈 单调栈维护递增区间,每次不满足 ...

  9. BZOJ1113 海报PLA1(单调栈入门题)

    一,自己思考下 1,先自己思考下 N个矩形,排成一排,现在希望用尽量少的海报去cover住它们. 2,不懂. 着实不懂. 3,分析下,最优性问题对吧,然后就每什么想法了.. 虽然肯定和单调栈和单调队列 ...

随机推荐

  1. Spring Aop切点

    切点用于准确定位应该在什么地方应用切面的通知.通知和切点是切面的最基本的元素.在Spring AOP中要使用AspectJ的切点表达式来定义切点.下面我们列出Spring AOP所支持的AspectJ ...

  2. PAT 天梯赛 L2-028. 秀恩爱分得快 【数据处理】

    题目链接 https://www.patest.cn/contests/gplt/L2-028 思路 0.只处理被询问的情侣的亲密度,否则会超时 1.要注意输入数字要用字符串,还要标记性别 因为 输出 ...

  3. linux 下ftp的安装配置 图文教程

    0.安装ftp的前置条件是关掉SElinux # vi /etc/selinux/config 修改 SELINUX=” disabled ” ,重启服务器.若相同,则跳过此步骤. 1. 可先查看是否 ...

  4. mini2440 最小根文件系统制作和nfs启动

    mini2440 内核启动后,可以用busybox制作一个简单的根文件系统并用nfs来启动该文件系统 启动mini2440, 按任意键进入uboot,按q键进入uboot命令行: 执行以下命令: se ...

  5. Android6.0 旋转屏幕(五)WMS启动应用流程(屏幕方向相关)

    一.强制设置方向 1.Activity 如果要强制设置一个Activity的横竖屏可以通过Manifest去设置,跟Activity相关的信息都会保存在ActivityInfo当中. android: ...

  6. 时间序列数据库——索引用ES、聚合分析时加载数据用什么?docvalues的列存储貌似更优优势一些

    加载 如何利用索引和主存储,是一种两难的选择. 选择不使用索引,只使用主存储:除非查询的字段就是主存储的排序字段,否则就需要顺序扫描整个主存储. 选择使用索引,然后用找到的row id去主存储加载数据 ...

  7. P2764 [网络流24题]最小路径覆盖问题[最大流]

    地址 这题有个转化,求最少的链覆盖→即求最少联通块. 设联通块个数$x$个,选的边数$y$,点数$n$个 那么有 $y=n-x$   即  $x=n-y$ 而n是不变的,目标就是在保证每个点入度.出度 ...

  8. ACM学习历程—HDU 4287 Intelligent IME(字典树 || map)

    Description We all use cell phone today. And we must be familiar with the intelligent English input ...

  9. x264源代码分析-转

    相关说明: 1.     使用版本:  x264-cvs-2004-05-11 2.     这次的分析基本上已经将代码中最难理解的部分做了阐释,对代码的主线也做了剖析,如果这个主线理解了,就容易设置 ...

  10. hibernate学习 六 Hibernate缓存

    缓存: 如果在集群环境下使用Hibernate时,(集群有节点A ,节点B) 当请求,发往A节点,A在数据库中修改了一条记录,然后节点B的缓存中如何实时的更新节点A修改的新数据          hi ...