Bentley的 Programming Pearls (第2版),在关于最大子阵列问题的章节中,描述了它的二维版本:
宾利提到,截至本书的出版日期(2000年),寻找最佳解决方案的问题是公开的。...我们得到了一个 n × n 实数数组,我们必须找到任何矩形子数组中包含的最大总和。这个问题的复杂性是什么?
答案 0 :(得分:3)
这个问题的一维解决方案(最大子阵列)是使用称为“Kadane算法”的算法的Theta(n)(我确信还有其他算法,但我对这个算法有个人经验)。这个问题的2D解决方案(最大子矩形)已知使用Kadane算法的实现O(n ^ 3)(我再次肯定还有其他算法,但我之前使用过它)。
虽然我们知道可以在Theta(n ^ 3)中找到2D解,但是没有人能够证明n ^ 3是否是下限。对于许多算法,当您增加维度时,您会将算法的下限增加一个恒定的幅度。由于这个特殊的问题,复杂性并没有线性增加,因此没有已知的解决方案适用于任何给定的维度,因此问题仍然存在。
参考类似的情况:我们知道2个矩阵可以在n ^ 3时间内相乘。还有一种算法可以在n ^ 2.8中完成,我相信。但是,没有数学表明我们不能低于n ^ 2.8,所以它仍然是一个“开放”算法。
答案 1 :(得分:1)
// Program to find maximum sum subarray in a given 2D array
#include <stdio.h>
#include <string.h>
#include <limits.h>
#define ROW 4
#define COL 5
// Implementation of Kadane's algorithm for 1D array. The function returns the
// maximum sum and stores starting and ending indexes of the maximum sum subarray
// at addresses pointed by start and finish pointers respectively.
int kadane(int* arr, int* start, int* finish, int n)
{
// initialize sum, maxSum and
int sum = 0, maxSum = INT_MIN, i;
// Just some initial value to check for all negative values case
*finish = -1;
// local variable
int local_start = 0;
for (i = 0; i < n; ++i)
{
sum += arr[i];
if (sum < 0)
{
sum = 0;
local_start = i+1;
}
else if (sum > maxSum)
{
maxSum = sum;
*start = local_start;
*finish = i;
}
}
// There is at-least one non-negative number
if (*finish != -1)
return maxSum;
// Special Case: When all numbers in arr[] are negative
maxSum = arr[0];
*start = *finish = 0;
// Find the maximum element in array
for (i = 1; i < n; i++)
{
if (arr[i] > maxSum)
{
maxSum = arr[i];
*start = *finish = i;
}
}
return maxSum;
}
// The main function that finds maximum sum rectangle in M[][]
void findMaxSum(int M[][COL])
{
// Variables to store the final output
int maxSum = INT_MIN, finalLeft, finalRight, finalTop, finalBottom;
int left, right, i;
int temp[ROW], sum, start, finish;
// Set the left column
for (left = 0; left < COL; ++left)
{
// Initialize all elements of temp as 0
memset(temp, 0, sizeof(temp));
// Set the right column for the left column set by outer loop
for (right = left; right < COL; ++right)
{
// Calculate sum between current left and right for every row 'i'
for (i = 0; i < ROW; ++i)
temp[i] += M[i][right];
// Find the maximum sum subarray in temp[]. The kadane() function
// also sets values of start and finish. So 'sum' is sum of
// rectangle between (start, left) and (finish, right) which is the
// maximum sum with boundary columns strictly as left and right.
sum = kadane(temp, &start, &finish, ROW);
// Compare sum with maximum sum so far. If sum is more, then update
// maxSum and other output values
if (sum > maxSum)
{
maxSum = sum;
finalLeft = left;
finalRight = right;
finalTop = start;
finalBottom = finish;
}
}
}
// Print final values
printf("(Top, Left) (%d, %d)\n", finalTop, finalLeft);
printf("(Bottom, Right) (%d, %d)\n", finalBottom, finalRight);
printf("Max sum is: %d\n", maxSum);
}
// Driver program to test above functions
int main()
{
int M[ROW][COL] = {{1, 2, -1, -4, -20},
{-8, -3, 4, 2, 1},
{3, 8, 10, 1, 3},
{-4, -1, 1, 7, -6}
};
findMaxSum(M);
return 0;
}
////////I found this program , hope it will help you
答案 2 :(得分:0)
仅供参考,本书的新版本有答案,但它太模糊了,我不知道它会带来什么。
无论如何,我会使用分而治之+动态编程来解决这个问题。让我们将MaxSum(x,y)定义为由N X N数组的左上角最大的矩形内的任何子阵列的最大总和,高度为y,宽度为x。 (所以问题的答案将在MaxSum(n-1,n-1))
MaxSum(x, y) is the max between:
1) MaxSum(x, y-1)
2) MaxSum(x-1, y)
3) Array[x, y] (the number in this N X N array for this specific location)
4) MaxEnding(x, y-1) + SUM of all elements from Array[MaxEndingXStart(x, y-1), y] to Array[x, y]
5) MaxEnding(x-1, y) + SUM of all elements from Array[x, MaxEndingYStart(x-1, y)] to Array[x, y]
MaxEnding(x,y-1)是包含数组[x,y-1]中#的任何子数组的最大总和。 同样,MaxEnding(x-1,y)是包含数组[x-1,y]中#的任何子数组的最大总和。 MaxEndingXStart(x,y-1)是子阵列的STARTING x坐标,其具有包含在Array [x,y-1]中的#的任何子阵列的最大总和。 MaxEndingYStart(x-1,y)是子数组的STARTING y坐标,它具有包含在数组[x-1,y]中的#的任何子数组的最大总和。
下面#4和#5中的2个和可以通过保持特定行中遇到的所有元素的总和,当您遍历每个列,然后减去2个总和以获得特定部分的总和来轻松计算。
要实现这一点,您需要采用自下而上的方法,因为您需要计算Max(x,y-1),Max(x-1,y),MaxEnding(x,y-1),和MaxEnding(x-1,y)..所以你可以在计算MaxEnding(x,y)时进行查找。
//first do some preprocessing and store Max(0, i) for all i from 0 to n-1.
//and store Max(i, 0) for all i from 0 to n-1.
for(int i =1; i < n; i++){
for(int j=1; j < n; j++) {
//LOGIC HERE
}
}