最大连续子数组和 动态规划_最大连续子序列和(算法)

2019/5/2修改

最近看到的一道编程题目:

有一个数组,如1, -5, 8, 3, -4, 15, -8,查找其中连续和最大的相邻串的值。在本例中,最大值为8 + 3 + -4 + 15 = 22.

这道题最容易想到的算法就是暴力搜索:

  1. 第一遍从数组第一个元素开始,找到它与后面每个元素之间的连续元素之和的最大值并记录下来;
  2. 第二遍从数组第二个元素开始,找到它与后面每个元素之间的连续元素之和的最大值,并与前一遍找到的最大值做比较,记录二者之中较大的值;
  3. 以此类推直到最后一个元素,便可以找到整个数组的最大连续子序列和。

下面看看go代码:

package 

这个算法简单粗暴易于理解,但效率不高,要找到最大连续子序列和一共需要n + (n - 1) + (n - 2) + ... + 1次访问数组元素,也就是时间复杂度是O(n*n)。那么有没有更好的算法呢?

我们来分析一下。

首先假设我们已经找到了最大连续和子串在数组中的起始位置(i)和结束位置(j),其中i <= j,即最大和maxSum = a[i] + a[i + 1] + ... + a[j],我们来看看这个子串有什么性质:

  1. a[i] > 0,否则我们完全可以去掉a[i]这个元素 而得到一个更大的和;
  2. i > 0且a[i - 1] < 0 或 i == 0,下面假设i > 0,这一条性质也是因为如果a[i - 1] > 0的话我们求和时可以加上a[i - 1]这个元素得到一个更大的和;
  3. 元素a[i - 1]与它之前的任一元素之间的子串之和sum < 0 ,即对于任何一个m(0 <= m < i - 1),则有a[m] + a[m + 1] + ... + a[i - 1] < 0,这条性质同样可以用反证法证明。

如果一时想不明白上面的第三条性质,可以用笔在纸上画画图帮助我们分析。根据第二三条性质,我们感觉 a[i - 1]是一个分界点,最大和的子串要么就在a[i - 1]元素之后,要么就在a[i - 1]之前,最大和的子串不可能跨过a[i - 1]这个点。读者可以仔细想一下这是为什么?还是可以用前面的反证法来思考。

下面举2个例子来看看:

第一个例子:假设数组为 1,-2, 3, 4,5,很容易发现-2这个元素满足前述的3个性质:
-2 本身是负数
-2 + 1 = -1 < 0
所以-2是这样一个分界点,最大和的字串要么在-2之后要么在之前,-2之前的和是1,之后的和sum = 3 + 4 + 5 = 12,所以这个字串的最大和为12。
我们稍微改变一下数组的元素就可以看到最大和字串在分解点之前的情况: 第二个例子:假设数组为 100,-101, 3, 4,5,很容易发现-101这个元素满足前述的3个性质:
-101 本身是负数
-101 + 100 = -1 < 0
所以-101是这样一个分界点,最大和的字串要么在-101之后要么在之前,-101之前的和是100,之后的和sum = 3 + 4 + 5 = 12,所以这个字串的最大和为100。

根据分析我们可以得出结论:

  • 只要找到分解点 a[i - 1],最大和的子串要么就在a[i - 1]元素之后,要么就在a[i - 1]之前,最大和的子串不可能跨过a[i - 1]这个点;
  • 一个数组中可能有多个这种分界点,但每个分界点都可以把前后完全分开,可以单独算分界点之间的最大和,然后在这些最大和之间取最大值。

假设对于数组a,我们找到了两个分界点a[i]和a[j],那么整个数组的最大字串和就是max(sum(a[0]...a[i-1]), sum(a[i+1]...a[j-1]), sum(a[j+1]...a[len-1]))

那么怎么去找这个分界点呢?

我们从前面的第3个性质可以看出如果a[i-1]是分界点,那么a[0]到a[i - 1]之和必定为负数,所以我们就从a[0]开始逐个往后求和,为了便于描述我们把这个和记为sum,sum第一次变成负数时就是我们要找的分界点。可能您会说sum(a[0]...a[i-1])<0并不代表sum(a[m]...a[i-1])<0 (m < i -1)呀?看看找这个分界点的方法,我们是从第一个元素开始求和,分界点是当sum第一次变成负数时找到的元素,也就是说a[0]到a[m-1]之和必定大于0,记为sum1, a[m]到a[i-1]之和记为sum2, 于是有关系sum1 + sum2 = sum < 0 推出sum2 = sum - sum1 < 0.

分析到这里算法基本上就出来了,下面给出go代码:

func 
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
最大连续序列和问题是指在一个序列中,找到一个连续序列,使得它们的和最大。这个问题可以使用分治法和动态规划算法来解决。 1. 分治法 使用分治法求解最大连续序列和问题的思路是:将原序列分成两个部分,最大序列可能存在于左半部分、右半部分或跨越左右两个部分。分别计算这三种情况下的最大序列和,然后取最大值即可。 下面是使用分治法求解最大连续序列和问题的Python代码示例: ```python def max_subarray(nums: List[int]) -> int: return _max_subarray(nums, 0, len(nums) - 1) def _max_subarray(nums: List[int], left: int, right: int) -> int: if left == right: return nums[left] mid = (left + right) // 2 left_max = _max_subarray(nums, left, mid) right_max = _max_subarray(nums, mid + 1, right) # 计算跨越左右两个部分的最大序列和 cross_max = nums[mid] left_cross_max = nums[mid] for i in range(mid - 1, left - 1, -1): left_cross_max += nums[i] cross_max = max(cross_max, left_cross_max) right_cross_max = nums[mid + 1] for i in range(mid + 2, right + 1): right_cross_max += nums[i] cross_max = max(cross_max, right_cross_max) return max(left_max, right_max, cross_max) ``` 在上面的代码中,我们使用递归的方式将原序列分成两个部分,然后计算跨越左右两个部分的最大序列和、左半部分的最大序列和、右半部分的最大序列和,取三者中的最大值作为整个序列的最大序列和。 2. 动态规划 使用动态规划算法求解最大连续序列和问题的思路是:从头开始遍历序列,对于每一个位置i,计算以i结尾的最大序列和,然后取所有的最大序列和的最大值作为问题的解。 具体来说,我们定义一个数组dp,其中dp[i]表示以i结尾的最大序列和。对于dp[i]来说,它的值可以由dp[i-1]和nums[i]计算得到,即dp[i] = max(dp[i-1] + nums[i], nums[i])。 下面是使用动态规划算法求解最大连续序列和问题的Python代码示例: ```python def max_subarray(nums: List[int]) -> int: dp = [0] * len(nums) dp[0] = nums[0] max_sum = nums[0] for i in range(1, len(nums)): dp[i] = max(dp[i-1] + nums[i], nums[i]) max_sum = max(max_sum, dp[i]) return max_sum ``` 在上面的代码中,我们使用一个循环遍历整个序列,计算以每一个位置为结尾的最大序列和,并且更新全局最大值。 希望这个回答能够帮到您!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值