【剑指offer】面试题31:连续子数组的最大和

题目:输入一个整型数组,数组里有正数也由负数。数组中一个或者连续的多个整数组成一个子数组。求所有子数组的和的最大值。要求时间复杂度为 O(n)。

例如输入的数组为{1,-2,3,10,-4,7,2,-5},其最大的子数组为{3,10,-4,7,2},因此所有子数组的和的最大值为 18。

 应用动态规划法:

我们需要保存两个值,一个存放遍历到当前位置的和kCurSum,另一个存放遍历到当前位置时的最大和kGreatestSum。假设现在我们遍历到第K个位置的元素:

1、如果当前的和kCurSum小于等于0,那么我们置当前和为遍历到的当前元素arr[k];

2、如果,kCurSum大于 0, 那么我们将 kCurSum 置为 kCurSum 与 arr[k] 的和;

3、将变化后的 kCurSum 与当前最大和 kGreatestSum 比较;

4、如果前者kCurSum 大于 后者kGreatestSum,将 kGreatestSum 置为 kCurSum;

4、否则,继续遍历;

 

完整代码如下:

 1 #include <iostream>
 2 #include <stdexcept>
 3 
 4 bool isAllElemsNegative(int *arr, int len);
 5 int findMaxNum(int *arr, int len);
 6 
 7 int findGreatestSumOfSubArray(int *arr, int len)
 8 {
 9     if(arr == NULL || len <= 0)
10         throw std::out_of_range("Invalid Input");
11 
12     int nMaxSum = 0;
13     int curSum = 0;
14 
15     if(isAllElemsNegative(arr, len))
16     {
17         return findMaxNum(arr, len);
18     }
19 
20     for(int i = 0; i < len; ++i) // 数组存在正数
21     {
22         if(curSum <= 0)
23             curSum = arr[i];
24         else
25             curSum += arr[i];
26 
27         if(curSum > nMaxSum)
28             nMaxSum = curSum;
29     }
30     return nMaxSum;
31 }
32 
33 bool isAllElemsNegative(int *arr, int len)
34 {
35     bool isNegative = true;
36 
37     for(int i = 0; i < len; ++i)
38     {
39         if(arr[i] >= 0)
40         {
41             isNegative = false;
42             break;
43         }
44     }
45     return isNegative;
46 }
47 
48 int findMaxNum(int *arr, int len)
49 {
50     int num = arr[0];
51     for(int i = 1; i < len; ++i)
52     {
53         if(arr[i] > num)
54             num = arr[i];
55     }
56     return num;
57 }
58 
59 int main(int argc, char *argv[])
60 {
61     // int arr[8] = {1, -2, 3, 10, -4, 7, 2, -5};
62     int arr[8] = {-3, -5, -2, -9, -12, -1, -90, -5};
63     int max = findGreatestSumOfSubArray(arr, 8);
64     std::cout << "The max Sum is: " << max << std::endl;
65 
66     return 0;
67 }
View Code

 

本文完。

posted @ 2015-07-07 21:00  Stephen_Hsu  阅读(473)  评论(0编辑  收藏  举报