C语言最大连续子序列和问题,分治法:求最大连续子序列和问题

题目:

最大连续子数列和一道很经典的算法问题,给定一个数列,其中可能有正数也可能有负数,我们的任务是找出其中连续的一个子数列(不允许空序列),使它们的和尽可能大。

例:序列{-10 1 2 3 4 -5 -23 3 7 -21},其最大的连续子序列为{1 2 3 4}或{3 7},最大和为10.

分析:分而治之。我们有一个很复杂的大问题,很难直接解决它,但是我们发现可以把问题划分成子问题,如果子问题规模还是太大,并且它还可以继续划分,那就继续划分下去。直到这些子问题的规模已经很容易解决了,那么就把所有的子问题都解决,最后把所有的子问题合并,我们就得到复杂大问题的答案了。可能说起来简单,但是仍不知道怎么做,接下来分析这个问题:

首先,我们可以把整个序列平均分成左右两部分,答案则会在以下三种情况中:

1、所求序列完全包含在左半部分的序列中。

2、所求序列完全包含在右半部分的序列中。

3、所求序列刚好横跨分割点,即左右序列各占一部分。

前两种情况和大问题一样,只是规模小了些,如果三个子问题都能解决,那么答案就是三个结果的最大值。我们主要研究一下第三种情况如何解决:

57cbaa2f84c5bccd0c63da8ddeb40a22.png

我们只要计算出:以分割点为起点向左的最大连续序列和、以分割点为起点向右的最大连续序列和,这两个结果的和就是第三种情况的答案。因为已知起点,所以这两个结果都能在O(N)的时间复杂度能算出来。

递归不断减小问题的规模,直到序列长度为1的时候,那答案就是序列中那个数字。综上所述,C语言代码如下,递归实现:#include

//N是数组长度,num是待计算的数组,放在全局区是因为可以开很大的数组

int N, num[16777216];

int solve(int left, int right)

{

//序列长度为1时

if(left == right)

return num[left];

//划分为两个规模更小的问题

int mid = left + right >> 1;

int lans = solve(left, mid);

int rans = solve(mid + 1, right);

//横跨分割点的情况

int sum = 0, lmax = num[mid], rmax = num[mid + 1];

for(int i = mid; i >= left; i--) {

sum += num[i];

if(sum > lmax) lmax = sum;

}

sum = 0;

for(int i = mid + 1; i <= right; i++) {

sum += num[i];

if(sum > rmax) rmax = sum;

}

//答案是三种情况的最大值

int ans = lmax + rmax;

if(lans > ans) ans = lans;

if(rans > ans) ans = rans;

return ans;

}

int main()

{

//输入数据

scanf("%d", &N);

for(int i = 1; i <= N; i++)

scanf("%d", &num[i]);

printf("%d\n", solve(1, N));

return 0;

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值