您的位置:首页 > 其它

《算法导论》读书笔记(三)——分治策略之和最大连续子序列

2013-09-07 18:35 495 查看

最大子数组问题

问题描述:

找出一个数组A的和最大非空连续子数组。

一个简单的C++实现:

#include <iostream>

using namespace std;

template <typename T>
T find_Max_Crossing_Subarray(T a[],int low, int high, int mid)
{
T max_Left = a[mid];
T max_Right = a[mid + 1];
T left = a[mid];
T right = a[mid + 1];
for(int i = mid - 1; i >= low; i--)
{
left += a[i];
max_Left = max_Left > left ? max_Left : left;
}
for(int i = mid + 2; i <= high; i++)
{
right += a[i];
max_Right = max_Right > right ? max_Right : right;
}
return max_Left + max_Right;
}

template <typename T>
T find_Max(T a[],int low, int high)
{
if(high > low)
{
int mid = (low + high) / 2;
T max1 = find_Max(a, low, mid);
T max2 = find_Max(a, mid + 1, high);
T max3 = find_Max_Crossing_Subarray(a, low, high, mid);
T max2or3 = max2 > max3 ? max2 : max3;
return max1 > max2or3 ? max1 : max2or3;
}
else
{
return a[low];
}
}

int main()
{
int a[15] = {13,-3,-25,20,-3,-16,-23,18,20,-7,12,-5,-22,15,-4};
cout << find_Max(a, 0, 15) << endl;
return 0;
}

分治最大子序列算法的递归式:



可以用主方法求得其时间界为O(nlogn)

线性时间界解法

分治法不一定做的最好,最大子序列有线性时间解法,简单的C++实现如下:

#include <iostream>
using namespace std;

template <typename T, int N>
T find_Max(T (&a)
)
{
T max = a[0];
T sum = a[0];
for(int i = 1; i < N; i++)
{
if(sum < 0)
{
sum = 0;
}
sum += a[i];
if(max < sum)
{
max = sum;
}
}
return max;
}

int main()
{
int a[16] = {13,-3,-25,20,-3,-16,-23,18,20,-7,12,-5,-22,15,-4,7};
cout << find_Max(a) << endl;
return 0;
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: