二维最大子阵列

时间:2011-03-26 21:52:02

标签: algorithm optimization max arrays

Bentley的 Programming Pearls (第2版),在关于最大子阵列问题的章节中,描述了它的二维版本:​​

  

...我们得到了一个 n × n 实数数组,我们必须找到任何矩形子数组中包含的最大总和。这个问题的复杂性是什么?

宾利提到,截至本书的出版日期(2000年),寻找最佳解决方案的问题是公开的。
还是这样吗?哪个是最知名的解决方案?是否有指向近期文献的指针?

3 个答案:

答案 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
   }
}