最简单的“最大子段和”问题

给定n个整数(可能为负数)a1,a2,……an。求ai,ai+1,……aj 其中i<=i<=j<=n的子段和的最大值。当所有整数均为负数时我们定义其最大子段和为0,且初始和终止位置也为0。例如:当(a1,a2,a3,a4,a5,a6)=(-2,11,-4,13,-5,-2)时,最大子段和为a2+a3+a4=20,i=2,j=4(下标从1开始)这个问题我们称之为“最大子段和问题”。

在考试一上,我们假定n<=200,今天我们把n的范围规定修改为n<=2000,你的任务是设计一个程序解决它。

Input

输入由两行,第一行为n;第二行为n个整数。

Output

输出也有两行,第一行为:“From=xxx,To=xxx”;第二行为:“MaxSum=xxxx”,参见样例。

Sample Input

6
-2 11 -4 13 -5 -2

Sample Output

From=2,To=4
MaxSum=20
#include<bits/stdc++.h>
using namespace std;
const int MAXN=100005;
int n;
long long a[MAXN],sum,ansum;
pair<int,int>ans;
int main()
{
    while(scanf("%d",&n)!=EOF)
    {
        for(int i=1;i<=n;++i)
        {
            scanf("%lld",&a[i]);
        }
        bool flag=false;
        for(int i=1;i<=n;++i)
        {
            if(a[i]>=0)flag=true;
        }
        if(!flag)
        {
            printf("From=0,To=0\nMaxSum=0\n");
            continue;
        }
        int pos=0;
        ansum=-1;
        sum=0;
        for(int i=1;i<=n;++i)
        {
            sum+=a[i];
            if(sum<0)
            {
                sum=0;
                pos=i;
            }
            else
            {

                if(sum>ansum)
                {
                    ansum=sum;
                    ans=make_pair(pos+1,i);
                }
                else if(sum==ansum)
                {
                    ans=min(ans,make_pair(pos+1,i));
                }
            }
        }
        printf("From=%d,To=%d\nMaxSum=%lld\n",ans.first,ans.second,ansum);
    }
    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值