算法导论(最大子数组问题-线性时间复杂度算法分析与实现)

这是算法导论4.1-5的练习题,原本的方法是分治法做的,时间复杂度为O(nlgn)。这里的方法是线性时间复杂度。题目如下:

使用如下思想为最大子数组问题设计一个非递归的,线性时间复杂度的算法。从数组左边界开始,由左至右处理,记录到目前为止已经处理过的最大子数组。若已知A[1...j]的最大子数组,基于如下性质将解扩展为A[1...j+1]的最大子数组:A[1...j+1]的最大子数组要么是A[1...j]的最大子数组,要么是某个子数组A[i...j+1](1<=i<=j+1)。在已知A[1...j]的最大子数组的情况下,可以在线性时间内找出形如A[i..j+1]的最大子数组。

由题意我们可以推导出:A[1...j+1]的最大子数组要么是A[1...j]的最大子数组(这里我们假设A[1...j]的最大子数组为A[left_num,right_num],sum=[left_num,right_num]),要么是A[i...j+1](right_numleft_num,j+1]。我们可以把[right_num+1,j+1]之间的元素用变量k隔开,使得sum1=[right_num+1,k-1]<0,sum2=[k,j+1]>0(A[k]>0,A[k-1]<0)。于是A[1...j+1]的最大子数组就是max(sum,sum2,sum+sum1+sum2);

代码如下:

#include
#define N 16
int main() {
	int left_num = 0, right_num = 0, sum1 = 0, sum2 = 0, k = 0;
		int A[] = {13,-3,-25,20,-3,-16,-23,18,20,-7,12,-5,-22,15,-4,7};
		int sum = A[0];
		for(int j = 1; j < N; j++) {
			sum2 += A[j];
			if(sum2 < 0) {
				sum1 += sum2;
				sum2 = 0;
				k = j + 1;
				continue;
		    }
			if(sum2 >= sum) {
				if(sum + sum1 >= 0) {
					right_num = j; k = j + 1; 
					sum += (sum1 + sum2);
					sum1 = 0; sum2 = 0; 
				}
				else {
					left_num = k; right_num = j; sum = sum2;
					k = j + 1; sum1 = 0; sum2 = 0; 
				}
				continue;
			}
			if(sum1 + sum2 >= 0) {
				right_num = j; k = j + 1; 
				sum += (sum1 + sum2);
				sum1 = 0; sum2 = 0;
			}
		}
		printf("Please output the element of the biggest subarray:");
		for(int i = left_num; i <= right_num; i++){
			printf("%8d",A[i]);
		}
		printf("\n");
		printf("Please output the sum of the biggest subarray: %d",sum);
}


你可能感兴趣的:(算法)