给你N×M大的矩阵,里面分别有字符‘F'和’R',要找到一个最大的只有‘F'的矩阵,不能包含有’R‘。N,M<=1000。

一开始的思路是单调栈来求最大矩形面积,因为没看清题目不能包含’R'字符,所以算出每行的‘F'字符个数然后单调栈就WA了。。

然后想到要从左边开始,算出连续的‘F'字符个数,然后又WA了。

因为还有右边,所以右边开始的’F'字符也处理一下,也是WA。

接着想到这种情形:

R F F F R

R F F F R

这枚举F的开始结束的话都是不行的,那么因为就两种字符。然后我又上一步一样只是求出R连续的大小,然后统计

lf左边开始‘F'的连续个数然后单调栈

rf右边开始’F'的连续个数然后单调栈

lr左边开始‘R'的连续个数然后单调栈

rr右边开始’R'的连续个数然后单调栈。

那么答案我就觉得是max(lf,rf,n*m-max(lr,rr))

然后也WA,因为我想到一个反例:

 R R R R R

 R F  F F R

 R F  F F R

 R F  F F R

 R R R R R

仔细观察这个样例,发现其实之前的做法很麻烦,我们最后得出的矩形一定是有左边的(废话),但是这个左边不确定,只要确定了直接用单调栈就出来了。

所以想到可以枚举K列,每次从第K列开始算。代码如下:

#include <iostream>
#include <fstream>
#include <stdio.h>
#include <string>
#include <string.h>
#include <algorithm>
using namespace std;

int n,m,arr[2333],stk[2333],w[2333];
char ma[2333][2333],ss[233];
long long ans;
int main() {
    int k;
    scanf("%d",&k);
    while(k--) {
        long long lr,rr,lf,rf;
        lr=rr=lf=rf=0;
        memset(stk,0,sizeof stk);
        ans=0;
        memset(w,0,sizeof w);
        scanf("%d%d",&n,&m);
        for(int i=1; i<=n; ++i) {
            for(int j=1; j<=m; ++j)
                scanf("%s",ss),ma[i][j]=ss[0];

        }
        for(int k=1; k<=m; ++k) {
            for(int i=1; i<=n; ++i) {
                int tmp=0;
                int j=k;
                while(j<=m&&ma[i][j++]!='R')
                    ++tmp;
                /*
                for(int j=1;j<=m;++j)
                    if(ma[i][j]=='F')
                        ++tmp;
                */
                arr[i]=tmp;
            }
            memset(stk,0,sizeof stk);
            memset(w,0,sizeof w);
            lf=0;
            arr[n+1]=0;
            int p=0;
            for(int i=1; i<=n+1; ++i) {
                if(arr[i]>stk[p])
                    stk[++p]=arr[i],w[p]=1;
                else {
                    int wid=0;
                    while(stk[p]>arr[i]) {
                        wid+=w[p];
                        lf=max(lf,(long long)wid*stk[p]);
                        --p;
                    }
                    stk[++p]=arr[i];
                    w[p]=wid+1;
                }
            }
            ans=max(ans,lf);
        }
        printf("%lld\n",ans*3);
    }
    return 0;
}

 

posted on 2018-04-29 09:40  chagin  阅读(229)  评论(0编辑  收藏  举报