首先dp处理出每个位置在一行上的往后延伸的最长距离。
然后一列列的做。相当于求一个极大子矩形。可以用单调栈来做。
类似这题:portal
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <utility>
#include <stack>
using namespace std;
#define ll long long
#define inf 0x3f3f3f3f
#define N 2010
#define pa pair<int,int>
inline char gc(){
static char buf[1<<16],*S,*T;
if(S==T){T=(S=buf)+fread(buf,1,1<<16,stdin);if(T==S) return EOF;}
return *S++;
}
inline int read(){
int x=0,f=1;char ch=gc();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=gc();}
while(ch>='0'&&ch<='9') x=x*10+ch-'0',ch=gc();
return x*f;
}
int n,m,a[N][N],f[N][N],ans1,ans2;
int main(){
// freopen("a.in","r",stdin);
n=read();m=read();
for(int i=1;i<=n;++i){
for(int j=1;j<=m;++j) a[i][j]=read();int sum=0;
for(int j=m;j>=1;--j){
if(a[i][j]==a[i][j+1]) sum=0;++sum;f[i][j]=sum;
}
}for(int j=1;j<=m;++j){
stack<pa>q;
for(int i=1;i<=n;++i){
int tmp=0;
while(!q.empty()&&q.top().first>f[i][j]){
int x=q.top().first,y=q.top().second+tmp;q.pop();
ans2=max(ans2,x*y);x=min(x,y);ans1=max(ans1,x*x);tmp=y;
}q.push(make_pair(f[i][j],tmp+1));
if(i==n||a[i][j]==a[i+1][j]){
int tmp=0;
while(!q.empty()){
int x=q.top().first,y=q.top().second+tmp;q.pop();
ans2=max(ans2,x*y);x=min(x,y);ans1=max(ans1,x*x);tmp=y;
}
}
}
}printf("%d\n%d\n",ans1,ans2);
return 0;
}