给定n(1<=n<=100000)个整数(可能为负数)组成的序列a[1],a[2],a[3],…,a[n],求该序列如a[i]+a[i+1]+…+a[j]的子段和的最大值。当所给的整数均为负数时定义子段和为0,依此定义,所求的最优值为: Max{0,a[i]+a[i+1]+…+a[j]},1<=i<=j<=n。 例如,当(a[1],a[2],a[3],a[4],a[5],a[6])=(-2,11,-4,13,-5,-2)时,最大子段和为20.
第一种 暴力求解 最费时 时间复杂度O(n^3)
int maxsum(int a[],int n)
{
int sum , maxsum;
sum = maxsum = 0;
int i ,j , k;
for(i = 0;i
for(j = i;j
sum = 0;
for(k = i;k<=j;k++)
sum+=a[k];
if(sum>maxsum)
maxsum = sum;
}
}
return maxsum;
}
第二种 在第一种上的改进版 时间复杂度O(n^2)
int maxsum(int a[],int n)
{
int sum , maxsum;
sum = maxsum = 0;
int i ,j ;
for(i = 0;i
sum = 0;
for(j = i;j
sum += a[j];
if(sum>maxsum)
maxsum = sum;
}
}
return maxsum;
}
第三种 分而治之 递归思想
将数组一分为二,左边右边,整个数组最大的和可能是左半边最大和 也可能是右半边最大和 还有可能是跨越边界最大和 , 分别求出来 ,在比较大小。时间复杂度O(nlogn)
int max(int a[],int left,int right)
{
count++;
int sum,i, ret,center,leftmax,rightmax,left_max,right_max;
if(right==left)
return a[left]>0?a[left]:0;
center =(left+right)/2;
leftmax = max(a,left,center);
rightmax = max(a,center+1,right);
sum = left_max = 0;
for( i = center;i>=left;i--)
{
sum+=a[i];
if(sum>left_max)
left_max = sum;
}
sum = 0;
right_max = 0;
for(i= center+1;i<=right;i++)
{
sum+=a[i];
if(sum>right_max)
right_max = sum;
}
ret = left_max+right_max;
if(ret
if(ret
return ret;
}
第四种 动态规划 在线处理 时间复杂度O(n) 最快
int max(int a[],int n)
{
int sum,maxsum;
int i ;
sum = maxsum = 0;
for(i = 0;i
sum +=a[i];
if(sum>maxsum)
maxsum = sum;
else if(sum<0)
sum = 0;
}
return maxsum;
}