[编程之美] 2.14 求数组的子数组之和的最大值

问题描述:给定一个包含N个整数的数组,求数组的子数组之和的最大值。

这是递归和贪心策略的一个经典问题。现在,对这个问题进行一下总结。

1 明确题意

题目中的子数组要求是连续的,也就是数组中的某个连续部分。

如果数组中都是正整数,直接相加就行。因此,主要是要考虑负数的情况。

2 直接求所有的子数组和

最简单且容易理解的解法是求出所有的子数组和,然后保存最大的和。

int MaxSum(int *A, int n)
{
    int maximum = -INF;
    int sum = 0;
    int i = 0, j = 0;

    for(i = 0; i < n; ++i) {

        sum = 0;

        for(j = i; j < n; ++j) {

            sum += A[j];

            if(sum > maximum) {
                maximum = sum;
            }
        }
    }

    return maximum;
}

3 递归

将数组分成长度相等的两部分:A[0]~A[n/2-1]和A[n/2]~A[n-1]。

那么对原数组而言,子数组和最大的子数组要么出现在A[0]~A[n/2-1]中,要么出现在A[n/2]~A[n-1],要么子数组包含A[n/2-1]和A[n/2]并向左右延伸。

这就变成了递归:

#include <iostream>
#include <algorithm>
#include <limits>
#include <vector>
using namespace std;

int MaxSum(vector<int>::iterator beg, vector<int>::iterator end)
{
    if(end <= beg) {
        return 0;
    }

	if(beg + 1 == end) {
		return *beg;
	}

	vector<int>::difference_type len = end - beg;
	vector<int>::iterator mid = beg + len / 2;

	int left = MaxSum(beg, mid - 1);
	int right = MaxSum(mid, end);
	int sum1 = 0, maximum = numeric_limits<int>::min();

	for(vector<int>::iterator iter = mid - 1; iter != beg - 1; --iter) {

		sum1 += *iter;

		if(sum1 > maximum) {
			maximum = sum1;
		}
	}
    sum1 = maximum;

	int sum2 = 0;
    maximum = numeric_limits<int>::min();
	for(vector<int>::iterator iter = mid; iter != end; ++iter) {

		sum2 += *iter;

		if(sum2 > maximum) {
			maximum = sum2;
		}
	}
    sum2 = maximum;

	return max(max(left, right), sum1 + sum2);
}

int main(int argc, char const *argv[])
{
	vector<int> vec;

	vec.push_back(0);
	vec.push_back(-2);
	vec.push_back(3);
	vec.push_back(5);
	vec.push_back(-1);
	vec.push_back(2);

	cout << MaxSum(vec.begin(), vec.end()) << endl;

	return 0;
}

4 动态规划

假设已知A[0]~A[n-1]的子数组的和的最大值为m[n-1],而以A[n-1]结束的子数组的和的最大值为s[n-1],于是有:

(1)以A[n]为结束的子数组要么包含A[n-1]要么不包含A[n-1],如果包含A[n-1],那么以A[n]结束的子数组的和的值就是s[n - 1] + A[n],如果不包含A[n - 1],那么A[n]就是一个重新开始计算的地方,即当前只包含A[n],于是有:s[n] = max(A[n], A[n] + s[n-1]);

(2)同样有,A[0]~A[n]的子数组要么以A[n]结束要么不以A[n]结束,如果以A[n]结束,子数组的和就是s[n],如果不以A[n]结束,子数组的和就是m[n - 1],于是有:m[n] = max(s[n], m[n - 1]);

根据上面的两个递推式就有:

int MaxSum(vector<int>::iterator beg, vector<int>::iterator end)
{
    if(end <= beg) {
        return 0;
    }

    vector<int>::difference_type len = end - beg;
    vector<int> m, s;
    m.resize(len);
    s.resize(len);
    s[0] = *beg;
    m[0] = *beg;

    vector<int>::iterator m_iter = m.begin() + 1, s_iter = s.begin() + 1;
    for(vector<int>::iterator iter = beg + 1; iter != end; ++iter) {
        *s_iter = max(*iter, *iter + *(s_iter - 1));
        *m_iter = max(*s_iter, *(m_iter -1 ));
        ++m_iter;
        ++s_iter;
    }

    return m.back();
}

再来好好看看上面两个递推式:

s[n] = max(A[n], A[n] + s[n-1]);

m[n] = max(s[n], m[n - 1]);

从这两个递推式可以看出,如果已知s[n-1]、m[n-1]和A[n]就可以求得s[n]和m[n],因此,事实上,只需要用两个变量来保存中间结果,而不需要用两个数组。

代码与上面的基本类似,就不再列出来了。

5 贪心

同时,从上面的两个递推式还可以看出,如果s[n-1] >= 0,就将A[n]放到当前考虑的子数组中,如果s[n-1] < 0,就从A[n]开始重新计算一个子数组。

这实际上使用的是贪心策略,不过贪心策略要经过严格的证明,这里还是可以从动态规划的方式来理解。于是有得到了史上最精简代码:

int MaxSum(vector<int>::iterator beg, vector<int>::iterator end)
{
    if(end <= beg) {
        return 0;
    }

    int maximum = numeric_limits<int>::min();
    int cur = 0;

    for(vector<int>::iterator iter = beg; iter != end; ++iter) {

        if(cur >= 0) {
            cur += *iter;
        }
        else {
            cur = *iter;
        }

        if(cur > maximum) {
            maximum = cur;
        }
    }

    return maximum;
}

如果当前计算的子数组和>=0,就将当前遍历的数组成员加到里面;如果当前计算的子数组和<0,就从当前遍历的数组成员开始计算。

6 小结

本文给出了求子数组之和的最大值的四种方法,前面两种方法容易思考,代码略长且时间复杂度较高;后面两种方法不容易理解,但是代码简单且时间复杂度低。

其中的关键是先引出s[n]和m[n],得到子数组之和的最大值的递推式,然后再通过递推式来简化算法。

[编程之美] 2.14 求数组的子数组之和的最大值

时间: 2024-11-05 19:03:14

[编程之美] 2.14 求数组的子数组之和的最大值的相关文章

编程之美 2.14求数组的子数组之和的最大值

对于一个有N个元素的数组,a[0]~a[n-1],求子数组最大值. 如:数组A[] = [−2, 1, −3, 4, −1, 2, 1, −5, 4],则连续的子序列[4,−1,2,1]有最大的和6. 方法一:暴力 循环遍历,输出所有,判断最大的和 1 #include"iostream" 2 #define MAX 1001 3 using namespace std; 4 5 int main(){ 6 int n, a[MAX], sum , maxsum ; 7 8 cin &

编程之美2.14 求数组的子数组之和的最大值

问题描述: 一个有N个整数元素的一维数组(A[0], A[1], A[2],...,A[n-1]),这个数组当然有很多子数组,那么子数组之和的最大值是什么呢? 解法: 1. 暴力解法-------O(N^3) 2. 改进版暴力解法-------O(N^2) *3. 分治算法-------O(NlogN)(暂时未去实现) 4. 数组间关系法-------O(N) 具体思路和代码: 1.暴力解法 思路:Sum[i,...,j]为数组第i个元素到第j个元素的和,遍历所有可能的Sum[i,...,j].

编程之美之2.14 求数组的子数组之和的最大值

[题目] 一个有N个整数元素的一维数组(A[0],A[1],A[2],...A[n-1]),这个数组中当然有很多子数组,那么子数组之和的最大值是多少? 该子数组是连续的. 我们先来明确一下题意: (1)子数组意味着是连续的. (2)题目只需要求和,并不需要返回子数组的具体位置. (3)数组的元素是整数,所以数组可能包含正整数,负整数或者零. 举几个例子: 数组:[1,-2,3,5,-3,2]返回8 数组:[0,-2,3,5,-1,2]返回9 数组:[-9,-2,-3,-5,-3]返回8 [解法一

2.14 求数组的子数组之和的最大值

题目:给定一个一维数组,求这个数组的子数组之和的最大值. 最佳方法:动态规划! 一. 可以将一个大问题(N个元素数组)转化为一个较小的问题(N-1个元素数组).. 假设已经知道(A[1],...A[n-1])中最大的子数组的和为:All[1] 并且已经知道(A[1],...A[n-1])中包括A[1]的子数组的最大和为start[1] 所以最终的解All[0] = max(A[0], A[0]+start[1], All[1]) 所以通过DP来求解! 代码如下: #include <iostre

《编程之美》解题报告:2.14 求数组的子数组之和的最大值

引言 ? ? 本文用了五种方法来求解这道题,从最初的n^3的复杂度到后面的n的复杂度,分别用到了递推公式.分而治之以及动态规划的方法去一步步降低算法的复杂度. ? ? 解题报告 ? ? 首先我们很容易想到的一个解法就是三层遍历,首先子数组必定是连续的一串值,相当于从原数组array的下标范围0~n-1中选出i和j,去算arra[i]~array[j]的和,于是我们可以得到最初的第一个解法 ? ? public static int sol1(int[] array) { int lenght =

【编程之美】2.14 求数组的子数组之和的最大值

做过的,就说下思路. 用Sum记录A[0...N-1]中 以第i个元素结尾的子数组中的最大和, 若以第i-1个元素结尾的子数组小于0,那么以第i个元素结尾的子数组中的最大和就是 A[i]本身 否则是A[i] + Sum(i-1的) 总结起来就是 Sum = (Sum > 0) ? A[i] + Sum : A[i]: 再记录下所有出现过的Sum中最大的值就可以了. #include <stdio.h> int getMaxSubArraySum(int * a, int alen) {

第2章 数字之魅——求数组的子数组之和的最大值

求数组的子数组之和的最大值 问题描述 分析与解法 [解法一] 具体代码如下: 1 package chapter2shuzizhimei.maxsumsubarray; 2 /** 3 * 求数组的子数组之和的最大值 4 * [解法一] 5 * @author DELL 6 * 7 */ 8 public class MaxSumSubArray1 { 9 //求数组的子数组之和的最大值 10 public static double maxSum(double a[]){ 11 double

求数组的子数组之和的最大值?

自己写的代码考虑未周全,引入了额外的空间复杂度: //求数组的子数组之和的最大值 #include <iostream> #define N 12 using namespace std; int main() { //int a[]={-5,2,3,-3,-2,3,1,-5}; //int a[]={-5,2,0,3,-2,3,4,5}; int a[]={1,-2,3,10,-4,7,2,-5}; int flag,max,i,j=0; int sum[N]={0}; //(1)记录子数组

【编程之美】求数组的子数组之和的最大值

一个有N个整数元素的一维数组A[0],A[1],......,A[n-1],这个数组当然有很多子数组,那么子数组的最大值是什么呢? 分析与解法 我们先明确题意: 1. 题目说的子数组,是连续的: 2. 题目只需要求和,并不需要返回子数组的具体位置: 3. 数组中的元素是整数,所以数组可能包含有正整数.零.负整数: 4. 子数组不为空. 解法一:枚举 最简单的办法就是枚举所有的i和j,计算sum[i..j] = A[i]+A[i+1]+...+A[j],遍历所有可能的sum[i..j],找到最大值