给定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;
}