【JZOJ 5838】【广州市选2011一试】旅游路线 (前缀和)

版权声明:嗯随意转载吧,注明出处就好 https://blog.csdn.net/Fallen_Angel001/article/details/82047880

问题描述
GZOI队员们到X镇游玩。X镇是一个很特别的城镇,它有m+1条东西方向和n+1条南北方向的道路,划分成m*n个区域,这些区域标从北到南、从西到东的坐标标识为从坐标 (1,1) 到坐标(m,n)。 GZOI队员们预先对这m*n个区域打分V(i,j)(分数可正可负)。分数越高表示他们越想到那个地方,越低表示他们越不想去。为了方便游玩,队员们需要选定一个连续的区域集合作为活动范围。例如,如果他们选择了最西北的区域(m1,nl)和最东南(m2,n2)区域(m1<=m2,n1<=n2),那他们的活动范围是 {D(i,j)|m1<=i<=m2,n1<=j<=n2},其游览总分则为这些活动范围的区域总分。 GZOI队员们希望他们活动范围内的区域的分值总和最大。你的任务是编写一个程序,求出他们的活动范围(m1,nl),(m2,n2〉。
输入
输入第一行为整数m(1<=m<=200),n(1<=n<=200),用空格隔开 下面为m行,每行有n列整数,其中第i行第j列的整数,代表V(i,j),每个整数之间用空格隔开,每个整数的范围是 [-200000,200000],输入数据保证这些整数中,至少存在一个正整数。
输出
输出只有一行,为最高的分值。
样例输入
4 5
1 -2 3 -4 5
6 7 8 9 10
-11 12 13 14 -15
16 17 18 19 20
样例输出
146
算法讨论
预处理每行的前缀和,枚举从哪一列开始,行数从1到n的矩形,若分值小于零,就把ans清零接着做。

#include <cstdio>
#define MAX_N 206
#define LL long long
using namespace std;
int a[MAX_N][MAX_N],f[MAX_N][MAX_N],n,m;
LL s,Max;

int max(int a,int b)
{
    return a>b?a:b;
}

int main()
{
    scanf("%d%d",&n,&m);
    for (int i=1;i<=n;i++)
        for (int j=1;j<=m;j++)
        {
            scanf("%d",&a[i][j]);
            f[i][j]=f[i][j-1]+a[i][j];
        }
    for (int k=1;k<=m;k++)
        for (int j=k;j<=m;j++)
        {
            s=0;
            for (int i=1;i<=n;i++)
            {
                s+=(f[i][j]-f[i][k-1]);
                Max=max(s,Max);
                if (s<0)
                    s=0;
            }
        }
    printf("%lld",Max);
}
阅读更多
想对作者说点什么?

博主推荐

换一批

没有更多推荐了,返回首页