记录以上所有情况中的最大值即可
注意:观察方案寻找规律,递推注意顺序
#include
#include
int sum[1505][1505],s[1505][1505],ul[1505][1505],ur[1505][1505],dl[1505][1505],dr[1505][1505],l[1505],h[1505];//内存限制:不能定义成3000^2
int max(int a,int b)
{
if(a>b) return a;
return b;
}
int main()
{
int m,n,k,i,j,cur,ans=0;
scanf("%d%d%d",&m,&n,&k);
for(i=1;i<=m;i++)
for(j=1;j<=n;j++)
{
scanf("%d",&sum[i][j]);
sum[i][j]+=sum[i-1][j]-sum[i-1][j-1]+sum[i][j-1];//若最后减sum[i-1][j-1]可能爆int
if(i>=k&&j>=k) s[i][j]=sum[i][j]-sum[i-k][j]-sum[i][j-k]+sum[i-k][j-k];
}
for(i=k;i<=m;i++)
for(j=k;j<=n;j++)
ul[i][j]=max(max(ul[i-1][j],ul[i][j-1]),s[i][j]);
for(i=k;i<=m;i++)
for(j=n-k+1;j>=1;j--)//注意递推的顺序!!!
ur[i][j]=max(max(ur[i-1][j],ur[i][j+1]),s[i][j+k-1]);
for(i=m-k+1;i>=1;i--)
for(j=k;j<=n;j++)
dl[i][j]=max(max(dl[i][j-1],dl[i+1][j]),s[i+k-1][j]);
for(i=m-k+1;i>=1;i--)
for(j=n-k+1;j>=1;j--)
dr[i][j]=max(max(dr[i+1][j],dr[i][j+1]),s[i+k-1][j+k-1]);
for(i=k;i<=m;i++)
for(j=k;j<=n;j++)
{
l[i]=max(l[i],s[i][j]);
h[j]=max(h[j],s[i][j]);
}
for(i=k;i<=m-k;i++)//一横一竖
for(j=k;j<=n-k;j++)
{
ans=max(ans,ul[i][j]+ur[i][j+1]+dl[i+1][n]);
ans=max(ans,ul[i][n]+dl[i+1][j]+dr[i+1][j+1]);
ans=max(ans,ul[i][j]+dl[i+1][j]+ur[m][j+1]);
ans=max(ans,ul[m][j]+ur[i][j+1]+dr[i+1][j+1]);
}
for(i=2*k;i<=m-k;i++)//两条横线,枚举下面一条
ans=max(ans,ul[i-k][n]+l[i]+dl[i+1][n]);
for(i=2*k;i<=n-k;i++)//两条竖线,枚举右面一条
ans=max(ans,ul[m][i-k]+h[i]+ur[m][i+1]);
printf("%d",ans);
return 0;
}