zoukankan      html  css  js  c++  java
  • 贪心+栈+队列类瞎操作

    一.栈

    1. luogu P3467 [POI2008]PLA-Postering

      这个题听说是单调栈的入门题qwq 

      自己想的时候想到了用单调栈维护高度 依次入栈,while(即将入栈的高度小于栈顶),就弹出栈顶;

      但是具体怎么记答案还是没想到 于是又想了一个玄学做法,用桶记录所有出现过的高度 相邻两高度ans++; 后来发现这样没法记是否连贯 要记录就太麻烦了 不得不弃

      正解和第一个思路有关

      记录答案初始ans=n;  依旧处理栈顶,  如果栈顶==w ans--;

      因为:1.对于高度不同的两个 必须用两张覆盖  2.高度相同可以用一张拉直覆盖

      所以说在弹出时 栈顶!=w 就不会使答案减少

      

     1 #include<iostream>
     2 #include<cstdio>
     3 #include<algorithm>
     4 #include<cstring>
     5 #include<cmath>
     6 #include<stack>
     7 #define N 250100
     8 using namespace std;
     9 stack<int>st;
    10 int d[N],w[N];
    11 int n,ans;
    12 int main()
    13 {
    14     scanf("%d",&n); ans=n;
    15     for(int i=1;i<=n;i++)
    16     {
    17         scanf("%d%d",&d[i],&w[i]);
    18         while(!st.empty()&&st.top()>=w[i])
    19         {
    20             if(st.top()==w[i]) ans--;
    21             st.pop();
    22         }
    23         st.push(w[i]);
    24     }
    25     printf("%d",ans);
    26     return 0;
    27 }
    3467

     2. luoguP4147 玉蟾宫

      听说这还带一个最大子矩阵的模板

      然而蒟蒻的我只想到了n^3的暴力枚举

      首先预处理: 对于每一行向上处理可以走的步数     if(x=='F') pos[i][j]=pos[i-1][j]+1;

      然后对每一行 x 维护一个单调栈 记录 pos[x][i] 和 可延展的宽度

      如果 st.top().h>pos[x][i] 就弹栈 并处理更新答案

      最后注意的是输入字符串的1mol多空格 用getchar()解决

      一个很好的blog

     1 #include<iostream>
     2 #include<cstdio>
     3 #include<cstring>
     4 #include<cmath>
     5 #include<algorithm>
     6 #include<stack>
     7 #include<cctype>
     8 #define ll long long
     9 #define N 2010
    10 using namespace std;
    11 struct node
    12 {
    13     int h,w;
    14 }a[N];
    15 stack<node> st;
    16 int n,m;
    17 int ans,anss;
    18 int pos[N][N];
    19 void red(int &x)
    20 {
    21     int f=1;x=0;char c=getchar();
    22     while(!isdigit(c)) {if(c=='-') f=-1; c=getchar();}
    23     while(isdigit(c)) {x=x*10+c-'0'; c=getchar();}
    24     x*=f;
    25 }
    26 char get()
    27 {
    28     char c=getchar();
    29     while(c!='F'&&c!='R') c=getchar();
    30     return c;
    31 }
    32 void cal(int x)
    33 {
    34     int tmp;
    35     st.push((node){pos[x][1],1});
    36     for(int i=2;i<=m;i++)
    37     {
    38         tmp=0;
    39         while(!st.empty()&&st.top().h>=pos[x][i])
    40         {
    41             tmp+=st.top().w;
    42             ans=max(ans,st.top().h*tmp);
    43             st.pop();
    44         }
    45         st.push((node){pos[x][i],tmp+1});
    46     }
    47     tmp=0;
    48     while(!st.empty())
    49     {
    50         tmp+=st.top().w;
    51         anss=max(anss,st.top().h*tmp);
    52         st.pop();
    53     }
    54     ans=max(ans,anss);
    55 }
    56 int main()
    57 {
    58     red(n); red(m);
    59     for(int i=1;i<=n;i++)
    60         for(int j=1;j<=m;j++)
    61         {
    62             char x=get();
    63             if(x=='F') pos[i][j]=pos[i-1][j]+1;
    64         }
    65     for(int i=1;i<=n;i++) cal(i);
    66     printf("%d",ans*3);
    67     return 0;
    68 }
    4147

     3. luogu P3668 [USACO17OPEN]Modern Art 2

      一来终于反应觉得是个栈 但始终感觉不对:每一次我们要找的距离最远的配对的两个数 所以差点掉到想deque的坑里

      后来索性放弃想数据结构 考虑裸贪心 

      想到记录每个颜色出现的第一次和最后一次位置

      但如何维护呢? 不由得又回到了想栈 但是怎么维护进栈的次序达到一层一层处理呢?

      然后我就凉了

      正解高明之处在于:每次循环准备进栈的时候, i 直接跳过上一次进栈的颜色出现的最后位置 相当于 保证每次考虑的都是同层的颜色

      且考虑完同层后直接按顺序进入下一层

      实现如下:

    for(int i=1;i<=n;i++)
    {
        .........
        st.push((node){i,last[a[i]],1});
        i=last[a[i]];
    }

      还有一点在于记录答案,每次遇到区间内满足的点 (即层数+1)于是入栈的时间也累加1,即 tmp.t+1

     1 #include<bits/stdc++.h>
     2 #define N 200100
     3 using namespace std;
     4 struct node
     5 {
     6     int x,y,t;
     7 }tmp;
     8 stack<node>st;
     9 int n;
    10 int a[N],last[N];
    11 int main()
    12 {
    13     scanf("%d",&n);
    14     for(int i=1;i<=n;i++)
    15     {
    16         scanf("%d",&a[i]);
    17         last[a[i]]=i;
    18     }
    19     for(int i=1;i<=n;i++)
    20     {
    21         if(!a[i]) continue;
    22         st.push((node){i,last[a[i]],1});
    23         i=last[a[i]];
    24     }
    25     int ans=0;
    26     while(!st.empty())
    27     {
    28         tmp=st.top();
    29         ans=max(ans,tmp.t);
    30         st.pop();
    31         for(int i=tmp.x+1;i<=tmp.y-1;i++)
    32         {
    33             if(a[i]==a[tmp.x]) continue;
    34             if(!a[i])
    35             {
    36                 printf("-1");
    37                 return 0;
    38             }
    39             if(a[i]!=a[tmp.x]&&last[a[i]]>last[a[tmp.x]])
    40             {
    41                 printf("-1");
    42                 return 0;
    43             }
    44             st.push((node){i,last[a[i]],tmp.t+1});
    45             i=last[a[i]];
    46         }
    47     }
    48     printf("%d",ans);
    49     return 0;
    50 }
    3668
  • 相关阅读:
    查询AD账号的SID
    Linux下NFS搭建实验
    定制windows环境下cmd替代软件ConEmu
    [转载]硬盘MBR详细介绍
    HP_UX扩dump空间
    zero和null以及sparse
    Cisco MDS9222i光纤交换机最常用排错命令
    Brocade SAN Switch上简单配置AG
    RHEL环境下调试Shell脚本时遇到字符串转换整数的问题
    Brocade SAN SWITCH配置文件导出和恢复
  • 原文地址:https://www.cnblogs.com/kylara/p/9913294.html
Copyright © 2011-2022 走看看