题解 AT4927 【[AGC033D] Complexity】
一个显然的想法是设
注意到答案至多为
如果是竖着切则转移显然为
如果横着切, 则有
Code:
#include<bits/stdc++.h>
using namespace std;
int n,m;
char ch[188][188];
int sum[188][188];
int dp[2][188][188][188];
bool eq(int x1,int y1,int x2,int y2)
{
int siz=(x2-x1+1)*(y2-y1+1);
int c1=sum[x2][y2]-sum[x2][y1-1]-sum[x1-1][y2]+sum[x1-1][y1-1];
return c1==siz||c1==0;
}
void init()
{
for(int i=1;i<=n;++i)
for(int j=1;j<=m;++j)
sum[i][j]+=sum[i][j-1];
for(int i=1;i<=n;++i)
for(int j=1;j<=m;++j)
sum[i][j]+=sum[i-1][j];
for(int i=1;i<=n;++i)
for(int j=i;j<=n;++j)
for(int l=1,r;l<=m;l=r)
{
r=l;
while(eq(i,l,j,r)&&r<=m)r++;
for(int p=l;p<r;++p)dp[0][i][j][p]=r-1;
if(l==r)dp[0][i][j][l]=l-1,r++;
}
}
int find(int c,int u,int d,int L)
{
int l=u,r=d-1,mid;
int res=0;
int v1,v2;
while(l<=r)
{
mid=(l+r)/2;
v1=dp[c][u][mid][L];
v2=dp[c][mid+1][d][L];
res=max(res,min(v1,v2));
if(v1<v2)r=mid-1;
else l=mid+1;
}
return res;
}
void DP(int c)
{
int nv,lv;
for(int i=1;i<=n;++i)
for(int j=i;j<=n;++j)
for(int l=1;l<=m;++l)
{
lv=dp[c^1][i][j][l];
if(lv!=m)
{
nv=dp[c^1][i][j][lv+1];
nv=max(nv,find(c^1,i,j,l));
}
else nv=m;
dp[c][i][j][l]=nv;
}
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=1;i<=n;++i)
{
scanf("%s",ch[i]+1);
for(int j=1;j<=m;++j)
if(ch[i][j]=='#')sum[i][j]=1;
}
init();
int ans=0;
for(int i=1;dp[(i&1)^1][1][n][1]<m;++i) DP(i&1),ans=i;
printf("%d\n",ans);
return 0;
}