zoukankan      html  css  js  c++  java
  • 浅谈单调栈 By cellur925

    这位dalao的单调栈文章很棒!我写的是他的题单233. http://www.cnblogs.com/COLIN-LIGHTNING/p/8474668.html

    一、单调栈的一般写法

        for(int i=1;i<=n;i++)
        {
            int x=0;
            scanf("%d",&x);
            while(x>=sta[top]&&top)
                top--;
            sta[++top]=x;
        }

    而各种各样繁杂的题目正是在这个基础上维护一些其他的信息。

    二、注意事项

    栈不能为空。要随时注意,否则RE。

    计数类可能会用到$longlong$。

    三、例题详解

    例题0 LIS(最长上升子序列)辣个$nlogn$的算法其实本质上就是单调栈...

    例题1 音乐会的等待

    N个人正在排队进入一个音乐会。人们等得很无聊,于是他们开始转来转去,想在队伍里寻找自己的熟人。队列中任意两个人A和B,如果他们是相邻或他们之间没有人比A或B高,那么他们是可以互相看得见的。

    写一个程序计算出有多少对人可以互相看见。(注意,是互相看见

    维护一个单调不增的栈。(因为两个之间没有比他们高的人时他们才能互相看见,中间隔等高个人时是可以互相看见的)

    (注意!脑子别晕!233,这是栈!不是队列!栈顶是最后加的!)

    当新来的人身高小于栈顶,直接进来。答案在栈非空的情况下加1,这算的是栈顶和当前元素这对。(满足相邻的条件)

    当新来的人身高大于栈顶,弹栈直到满足小于(或等于)。因为维护的单调递减的栈,而新生还很高,所以自然能看得见。这里每次弹栈都加上那个元素的个数(具体接下来会讲)。满足性质后,再和栈顶比较。

    当新来的人身高等于栈顶,弹栈直到小于。我们把和它相等的都弹走,但是要注意记上和它相等的数出现的次数,以便之后使用,因为相等,所以肯定看的见,为接下来记录了信息(解释了第二种情况)。满足性质后,再和栈顶比较。

    综上,我们看出每个元素最多入栈出栈一次,复杂度$O(N)$,每次入栈后,都要在栈不为空的情况下,答案++,因为栈顶(最外面的元素)一定能看见新来的元素。

    Code

     1 #include<cstdio>
     2 #include<algorithm>
     3 
     4 using namespace std;
     5 typedef long long ll;
     6 
     7 int n,top,x,num;
     8 struct node{
     9     int val,num;
    10 }sta[500090];
    11 ll ans;
    12 
    13 int main()
    14 {
    15     scanf("%d",&n);
    16     for(int i=1;i<=n;i++)
    17     {
    18         scanf("%d",&x);
    19         node p=(node){x,1};
    20         while(x>=sta[top].val&&top)
    21         {
    22             ans+=sta[top].num;
    23             if(sta[top].val==x) p.num+=sta[top].num;
    24             //虽然弹走了 记下  以后用得到 
    25             top--;
    26         }
    27         if(top) ans++;
    28         sta[++top]=p;
    29     }
    30     printf("%lld",ans);
    31     return 0;
    32 }
    View Code

    例题2 [POI2008]PLA-Postering

    Byteburg市东边的建筑都是以旧结构形式建造的:建筑互相紧挨着,之间没有空间.它们共同形成了一条长长的,从东向西延伸的建筑物链(建筑物的高度不一).Byteburg市的市长Byteasar,决定将这个建筑物链的一侧用海报覆盖住.并且想用最少的海报数量,海报是矩形的.海报与海报之间不能重叠,但是可以相互挨着(即它们具有公共边),每一个海报都必须贴近墙并且建筑物链的整个一侧必须被覆盖(意思是:海报需要将一侧全部覆盖,并且不能超出建筑物链,即不能盖着没有海报存在的地方)

    读完题后,我们很快就会发现,那个宽度的条件其实是没用的,我们只需要考虑高度就行了。设开始我们需要海报的数量等于矩形的数量,我们再一点点减。那么,我们便可以维护一个高度单调递增的栈,在弹栈过程中,若遇到和自己相等的高度,就将答案减去1,因为如图,我们可以横着盖,我们只需要再把高出来的(於出来的)用一张海报即可。

    Code

     1 #include<cstdio>
     2 #include<algorithm>
     3 
     4 using namespace std;
     5 
     6 int n,ans,top;
     7 int sta[300000];
     8 
     9 int main()
    10 {
    11     scanf("%d",&n);
    12     for(int i=1;i<=n;i++)
    13     {
    14         int y=0,x=0;
    15         scanf("%d%d",&y,&x);
    16         while(top&&x<=sta[top])
    17         {
    18             if(x==sta[top]) ans++;
    19             top--;
    20         }
    21         sta[++top]=x;
    22     }
    23     printf("%d",n-ans);
    24     return 0;
    25 }
    View Code

    以上都是一些比较简单的线性问题。接下来我们要看的是那些或者在二维上,或者掺杂入实际的面积,这些更复杂的问题。

    例题3 玉蟾宫 哎又要看Freda学姐和rainbowcat虐狗了

    其实这题还有悬线法,是求最大子矩阵的一个方法,这里就不再说了==(大坑)

    抽象一下本题的模型:有障碍点的情况下的最大子矩形。

    虽然这题是二维的了,但是我们还可以分别计算情况。

    首先,我们用一个$f[i][j]$表示在$(i,j)$点,前$i$行,第$j$列,以第$i$行结尾,连续的‘F’个数。(最大可延伸距离)

    行与行之间的问题就解决了。

    然后在同一行,不同列上的问题就可以用单调栈(单调递增栈)来维护了。

    单调栈中存储两个信息。此单位高度$height$,和对应可控宽度$wid$(对应可控我觉得说的非常准确)

    同理,我们当前的高度(f数组)大于栈顶时,直接把它压入栈。

    否则,就一直弹栈。弹栈的时候,我们要记录矩形的信息来更新答案。它的$wid$是所有弹栈元素的$wid$+1.(因为我们继承了之前的信息,这部分 到之后也是可以用的。因为被弹栈的元素高度均大于当前元素,在可控范围内。)

    之后更新答案即可。枚举到最后一列后,把栈都搞空。

    Code

     1 #include<cstdio>
     2 #include<algorithm>
     3 
     4 using namespace std;
     5 
     6 int n,m,ans,top,tmp,neww;
     7 struct node{
     8     int height,wid;
     9 }sta[2000];
    10 char qwq[5],mapp[2000][2000],f[2000][2000];
    11 
    12 void work(int x)
    13 {
    14     top=1;tmp=0;neww=0;
    15     sta[1].height=f[x][1];
    16     sta[1].wid=1;
    17     for(int i=2;i<=m;i++)
    18     {
    19         tmp=0;
    20         while(f[x][i]<=sta[top].height&&top)
    21         {
    22             tmp+=sta[top].wid;
    23             neww=max(neww,tmp*sta[top].height);
    24             top--;
    25         }
    26         sta[++top].height=f[x][i];
    27         sta[top].wid=tmp+1;
    28     }
    29     tmp=0;
    30     while(top)
    31     {
    32         tmp+=sta[top].wid;
    33         neww=max(neww,tmp*sta[top].height);
    34         top--;
    35     }
    36     ans=max(ans,neww);
    37 }
    38 
    39 int main()
    40 {
    41     scanf("%d%d",&n,&m);
    42     for(int i=1;i<=n;i++)
    43         for(int j=1;j<=m;j++) 
    44         {
    45             scanf("%s",qwq+1),mapp[i][j]=qwq[1];
    46             if(mapp[i][j]=='F') f[i][j]=f[i-1][j]+1;
    47         }
    48     for(int i=1;i<=n;i++) work(i);
    49     printf("%d",ans*3);
    50     return 0;
    51 }
    View Code

    例题4 Largest Rectangle in a Histogram

    在一条水平线上给出若干连续矩形,求包含于这些矩形的并集内部最大矩形的面积。

    这应该是单调栈的典型题目惹qwq。

    我们假设,到现在为止,读入的矩形高度都是递增的,那么如果突然读入了一个比上一个矩形矮的矩形,那么之前高的矩形搞出来就没有用了(於出来了)

    打叉的部分就没用了。没用就删去啊,所以我们在做的事情就是维护一个单调递增的矩形序列。与上题相似,弹矩形的时候,我们同样要维护弹出矩形的答案。

    其实这和上一题差不多啦qwq。

    Code

     1 #include<cstdio>
     2 #include<algorithm>
     3 #include<cstring>
     4 
     5 using namespace std;
     6 typedef long long ll;
     7 
     8 int n,top,x,tmp;
     9 ll ans;
    10 struct node{
    11     int height,wid;
    12 }sta[1000090];
    13 
    14 
    15 int main()
    16 {
    17     while(scanf("%d",&n)!=EOF&&n)
    18     {
    19         for(int i=1;i<=n;i++)
    20         {
    21             tmp=0;
    22             scanf("%d",&x);
    23             while(x<=sta[top].height&&top)
    24             {
    25                 tmp+=sta[top].wid;
    26                 ans=max(ans,1ll*tmp*sta[top].height);
    27                 top--;
    28             }
    29             sta[++top].height=x;
    30             sta[top].wid=tmp+1;
    31         }
    32         tmp=0;
    33         while(top)
    34         {
    35             tmp+=sta[top].wid;
    36             ans=max(ans,1ll*tmp*sta[top].height);
    37             top--;
    38         }
    39         printf("%lld
    ",ans);
    40         top=0;ans=0;
    41         memset(sta,0,sizeof(sta));
    42     }
    43     return 0;
    44 }
    View Code

    这种更新面积(我也不知道怎么总结)的问题,最后一定要记得把栈搞空啊qwq。

    单调栈先告一段落,下一次来看单调队列qwq。

  • 相关阅读:
    android中的AIDL进程间通信
    [Android自定义控件] Android自定义控件
    Android控件之SlidingDrawer(滑动式抽屉)详解与实例
    Android 广播大全 Intent Action 事件
    Activity和Service绑定
    String.format()用法
    Android之Handler用法总结
    android中的AIDL进程间通信
    Android: 在 TextView 里使用删除线
    fastjson生成和解析json数据
  • 原文地址:https://www.cnblogs.com/nopartyfoucaodong/p/9754215.html
Copyright © 2011-2022 走看看