记录以上所有情况中的最大值即可
注意:观察方案寻找规律,递推注意顺序
#include<stdio.h> #include<stdlib.h> 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; }