最大サブセグメント和(二)


最大サブセグメントと問題
 
解法二:分治法
アルゴリズム解析:
最大サブセグメントと3つの場所で表示される可能性があります.1)入力データの左半分2)全体が右半分に現れる.
3)入力データの中央部を越えて左右の2つの部分のうちの1つ前に位置する場合を再帰的に解くことができ,3つ目の場合の最大和は,前半の最大和(前半の最後の要素を含む)および後半の最大和(後半の最初の要素を含む)を求めることによって得られ,前後の2つの部分と加算される.3つの部分のうちの最大は、最大サブセグメント和である.
例:
前半後半
4  -3  5  -2    -1  2  6  -2
ここで、前半の最大サブセグメント和は6(A 1~A 3)であり、後半の最大サブセグメント和は8(A 6~A 7)である.
前半部はその最後の要素の最大和4(A 1~A 4)を含み、後半部はその最初の要素の最大和7(A 5~A 7)を含み、
この2つの部分にまたがる最大和は4+7=11である.
したがって、シーケンスの最大フィールド和は11である
 
アルゴリズムは以下のように実現される.
/*
 * description:      
 *     
 *             。
 * 1)             
 * 2)        。
 * 3)                     
 *            ,                       (             )
 *           (            )   ,        。
 *         ,       .
 * auther: cm
 * date: 2010/11/20
 */
public class MaxSubSumRec 
{
	private static int maxSumRec(int[] a, int left, int right)
	{
		if (left == right)
		{
			if (a[left] > 0)
			{
				return a[left];
			}
			else
			{
				return 0;
			}
		}

		int center = (left + right) / 2;
		int maxLeftSum = maxSumRec(a, left, center);
		int maxRightSum = maxSumRec(a, center + 1, right);

		int maxLeftBorderSum = 0;
		int leftBorderSum = 0;
		for (int i = center; i >= left; i--)
		{
			leftBorderSum += a[i];
			if (leftBorderSum > maxLeftBorderSum)
			{
				maxLeftBorderSum = leftBorderSum;
			}
		}

		int maxRightBorderSum = 0;
		int rightBorderSum = 0;
		for (int i = center + 1; i <= right; i++)
		{
			rightBorderSum += a[i];
			if (rightBorderSum > maxRightBorderSum)
			{
				maxRightBorderSum = rightBorderSum;
			}
		}

		return max(maxLeftSum, maxRightSum, maxRightBorderSum + maxLeftBorderSum);
	}

	//    
	public static int maxSubSum(int[] a)
	{
		return maxSumRec(a, 0, a.length - 1);
	}

	//        
	private static int max(int a, int b, int c)
	{
		int max = a > b ? a : b;
		max = c > max ? c : max;

		return max;
	}

	public static void main(String[] args) 
	{
		int[] a = {-2, 11, -4, 13, -5, -2};
		System.out.println(MaxSubSumRec.maxSubSum(a));
	}
}