首尾相连数组的最大子数组和


分类: C语言 算法 ACM   122人阅读  评论(0)  收藏  举报

目录(?)[+]

前言

这是昨晚参加九度oj的7月份月赛碰到的题目,很有意思,今早查了一下,原来《编程之美》中已经讲过类似题目,看来我把《剑指offer》看完之后,下一本书就是《编程之美》了,今天6点起床想看会英语,可惜还是搞算法了

题目

[html]  view plain copy
  1. 题目描述:  
  2. 给定一个由N个整数元素组成的数组arr,数组中有正数也有负数,这个数组不是一般的数组,其首尾是相连的。数组中一个或多个连续元素可以组成一个子数组,其中存在这样的子数组arr[i],…arr[n-1],arr[0],…,arr[j],现在请你这个ACM_Lover用一个最高效的方法帮忙找出所有连续子数组和的最大值(如果数组中的元素全部为负数,则最大和为0,即一个也没有选)。  
  3. 输入:  
  4. 输入包含多个测试用例,每个测试用例共有两行,第一行是一个整数n(1=<n<=100000),表示数组的长度,第二行依次输入n个整数(整数绝对值不大于1000)。  
  5. 输出:  
  6. 对于每个测试用例,请输出子数组和的最大值。  
  7. 样例输入:  
  8. 6  
  9. 1 -2 3 5 -1 2  
  10. 5  
  11. 6 -1 5 4 -7  
  12. 样例输出:  
  13. 10  
  14. 14  

这里以  arr[6] = {1, -2, 3, 5, -1, 2}为例做介绍

首尾不相连最大子数组和

这是典型的动态规划题目,假设sum[i]为前i个数据的最大子数组和,则得到状态转移方程为:




有了状态转移方程,代码很容易实现(c语言)

[cpp]  view plain copy
  1. /** 
  2.  * 首尾不相连 
  3.  */  
  4. int maxSumNoConnect(int *arr, int n)  
  5. {  
  6.     int i, max, *sum;  
  7.     sum = (int *)malloc(sizeof(int) * n);  
  8.   
  9.     sum[0] = max = arr[0];  
  10.     for (i = 1; i < n; i ++) {  
  11.         if (sum[i - 1] > 0)  
  12.             sum[i] = arr[i] + sum[i - 1];  
  13.         else  
  14.             sum[i] = arr[i];  
  15.   
  16.         if (sum[i] > max) max = sum[i];  
  17.     }  
  18.   
  19.     free(sum);  
  20.     return max;  
  21. }  

可以看到,这里我们用了O(n )的辅助数组sum,可以考虑直接简化辅助空间,代码如下:

[cpp]  view plain copy
  1. int maxSumNoConnectOne(int *arr, int n)  
  2. {  
  3.     int i, max, sum;  
  4.   
  5.     max = sum = arr[0] > 0 ? arr[0] : 0;  
  6.   
  7.     for (i = 1; i < n; i ++) {  
  8.         if ((arr[i] + sum) > 0) {  
  9.             sum += arr[i];  
  10.             max = max > sum ? max : sum;  
  11.         } else {  
  12.             sum = 0;  
  13.             max = max > arr[i] ? max : arr[i];  
  14.         }  
  15.     }  
  16.       
  17.     return max;  
  18. }  



首尾相连最大和子数组

如果数组是首尾相邻的,可以分成两种情况来讨论:

(1) 如果最大和子数组没有跨过首元素,则可以用原问题的解法来求解

(2)如果最大和子数组跨过了首元素,则可以继续分两种情况讨论:

        1. 如果数组元素全部为正,则所有元素之和就是最大值
        2. 如果数组中有负数,则最大和子数组必然不包括某些元素,而不被包括的元素中必然有最小和子数组,这样我们就可以先查找最小和子数组的最后一个点,并以下一个点为起点,利用(1)的方法循环遍历一遍数组,就可以得到最大和子数组了

图示如下:

首尾相连数组的最大子数组和_第1张图片


(ps:我靠,图片画歪了,算了,大家领会精神就好)

(1)可以用上面首尾不相连的做法求解

(2)的步骤如下

最小和子数组截止坐标

典型的动态规划思想,假设sum[i]为前i个数据的最小子数组和,则得到状态转移方程为:



实现代码(c语言)

[cpp]  view plain copy
  1. /** 
  2.  * 最小子数组和的截止坐标 
  3.  */  
  4. int indexInArrMinSum(int *arr, int n)  
  5. {  
  6.     int i, loc, min, *sum;  
  7.   
  8.     sum = (int *)malloc(sizeof(int) * n);  
  9.     min = sum[0] = arr[0];  
  10.     loc = 0;  
  11.   
  12.     for (i = 1; i < n; i ++) {  
  13.         if (sum[i - 1] > 0) {  
  14.             sum[i] = arr[i];  
  15.         } else {  
  16.             sum[i] = arr[i] + sum[i - 1];  
  17.         }  
  18.   
  19.         if (sum[i] < min) {  
  20.             loc = i;  
  21.             min = sum[i];  
  22.         }  
  23.     }  
  24.   
  25.     free(sum);  
  26.   
  27.     return loc;  
  28. }  

循环遍历

循环遍历只要从最小和子数组截止坐标下一个位置开始即可,无非遍历时需要注意取余而已

[cpp]  view plain copy
  1. /** 
  2.  * 首尾相连 
  3.  */  
  4. int maxSumConnect(int *arr, int n)  
  5. {  
  6.     int index, i, loc, cur, max, *sum;  
  7.   
  8.     sum = (int *)malloc(sizeof(int) * n);  
  9.   
  10.     index = indexInArrMinSum(arr, n);  
  11.       
  12.     max = sum[(index + 1) % n] = arr[(index + 1) % n];  
  13.   
  14.     for (i = 2; i < n; i ++) { // 不考虑最小子数组的截止坐标了  
  15.         loc = (index + i - 1) % n;  
  16.         cur = (index + i) % n;  
  17.         if (sum[loc] > 0) {  
  18.             sum[cur] = arr[cur] + sum[loc];  
  19.         } else {  
  20.             sum[cur] = arr[cur];  
  21.         }  
  22.           
  23.         if (sum[cur] > max)  
  24.             max = sum[cur];  
  25.     }  
  26.   
  27.     free(sum);  
  28.     return max;  
  29. }  

ac代码

[cpp]  view plain copy
  1. #include <stdio.h>  
  2. #include <stdlib.h>  
  3.    
  4. /** 
  5.  * 首尾不相连 
  6.  */  
  7. int maxSumNoConnect(int *arr, int n)  
  8. {  
  9.     int i, max, *sum;  
  10.     sum = (int *)malloc(sizeof(int) * n);  
  11.    
  12.     sum[0] = max = arr[0];  
  13.     for (i = 1; i < n; i ++) {  
  14.         if (sum[i - 1] > 0)  
  15.             sum[i] = arr[i] + sum[i - 1];  
  16.         else  
  17.             sum[i] = arr[i];  
  18.    
  19.         if (sum[i] > max) max = sum[i];  
  20.     }  
  21.    
  22.     free(sum);  
  23.     return max;  
  24. }  
  25.    
  26. /** 
  27.  * 最小子数组和的截止坐标 
  28.  */  
  29. int indexInArrMinSum(int *arr, int n)  
  30. {  
  31.     int i, loc, min, *sum;  
  32.    
  33.     sum = (int *)malloc(sizeof(int) * n);  
  34.     min = sum[0] = arr[0];  
  35.     loc = 0;  
  36.    
  37.     for (i = 1; i < n; i ++) {  
  38.         if (sum[i - 1] > 0) {  
  39.             sum[i] = arr[i];  
  40.         } else {  
  41.             sum[i] = arr[i] + sum[i - 1];  
  42.         }  
  43.    
  44.         if (sum[i] < min) {  
  45.             loc = i;  
  46.             min = sum[i];  
  47.         }  
  48.     }  
  49.    
  50.     free(sum);  
  51.    
  52.     return loc;  
  53. }  
  54.    
  55. /** 
  56.  * 首尾相连 
  57.  */  
  58. int maxSumConnect(int *arr, int n)  
  59. {  
  60.     int index, i, loc, cur, max, *sum;  
  61.    
  62.     sum = (int *)malloc(sizeof(int) * n);  
  63.    
  64.     index = indexInArrMinSum(arr, n);  
  65.        
  66.     max = sum[(index + 1) % n] = arr[(index + 1) % n];  
  67.    
  68.     for (i = 2; i < n; i ++) { // 不考虑最小子数组的截止坐标了  
  69.         loc = (index + i - 1) % n;  
  70.         cur = (index + i) % n;  
  71.         if (sum[loc] > 0) {  
  72.             sum[cur] = arr[cur] + sum[loc];  
  73.         } else {  
  74.             sum[cur] = arr[cur];  
  75.         }  
  76.            
  77.         if (sum[cur] > max)  
  78.             max = sum[cur];  
  79.     }  
  80.    
  81.     free(sum);  
  82.     return max;  
  83. }  
  84.    
  85. int main(void)  
  86. {  
  87.     int i, n, flag, conn_n, conn_y, *arr;  
  88.    
  89.     while (scanf("%d", &n) != EOF) {  
  90.         arr = (int *)malloc(sizeof(int) * n);  
  91.         for (i = 0, flag = 0; i < n; i ++) {  
  92.             scanf("%d", arr + i);  
  93.             if (*(arr + i) <= 0)  
  94.                 flag ++;  
  95.         }  
  96.    
  97.         // 均为负数返回0  
  98.         if (flag == n) {  
  99.             printf("0\n");  
  100.             continue;  
  101.         }  
  102.    
  103.         conn_n = maxSumNoConnect(arr, n);  
  104.         conn_y = maxSumConnect(arr, n);  
  105.         if (conn_n < conn_y)  
  106.             printf("%d\n", conn_y);  
  107.         else  
  108.             printf("%d\n", conn_n);  
  109.         free(arr);  
  110.     }  
  111.    
  112.     return 0;  
  113. }  
  114.    
  115.    
  116. /************************************************************** 
  117.     Problem: 1527 
  118.     User: wangzhengyi 
  119.     Language: C 
  120.     Result: Accepted 
  121.     Time:80 ms 
  122.     Memory:2084 kb 
  123. ****************************************************************/  

你可能感兴趣的:(算法,C语言,ACM)