array median transformation minimum steps

后端 未结 3 857
不思量自难忘°
不思量自难忘° 2021-01-13 02:12

Given an array A with n integers. In one turn one can apply the following operation to any consecutive subarray A[l..r] : assign to all A i (l <= i <= r) median of sub

3条回答
  •  一生所求
    2021-01-13 02:59

    The problem setter approach has exponential complexity. It is pretty good for N=30. But not so for larger sizes. I think, it's more interesting to find an exponential time solution. And I found one, with O(N4) complexity.

    This approach uses the fact that optimal solution starts with some group of consecutive maximal elements and extends only this single group until whole array is filled with maximal values.

    To prove this fact, take 2 starting groups of consecutive maximal elements and extend each of them in optimal way until they merge into one group. Suppose that group 1 needs X turns to grow to size M, group 2 needs Y turns to grow to the same size M, and on turn X + Y + 1 these groups merge. The result is a group of size at least M * 4. Now instead of turn Y for group 2, make an additional turn X + 1 for group 1. In this case group sizes are at least M * 2 and at most M / 2 (even if we count initially maximal elements, that might be included in step Y). After this change, on turn X + Y + 1 the merged group size is at least M * 4 only as a result of the first group extension, add to this at least one element from second group. So extending a single group here produces larger group in same number of steps (and if Y > 1, it even requires less steps). Since this works for equal group sizes (M), it will work even better for non-equal groups. This proof may be extended to the case of several groups (more than two).

    To work with single group of consecutive maximal elements, we need to keep track of only two values: starting and ending positions of the group. Which means it is possible to use a triangular matrix to store all possible groups, allowing to use a dynamic programming algorithm.

    Pseudo-code:

    For each group of consecutive maximal elements in original array:
      Mark corresponding element in the matrix and clear other elements
      For each matrix diagonal, starting with one, containing this element:
        For each marked element in this diagonal:
          Retrieve current number of turns from this matrix element
          (use indexes of this matrix element to initialize p1 and p2)
          p2 = end of the group
          p1 = start of the group
          Decrease p1 while it is possible to keep median at maximum value
          (now all values between p1 and p2 are assumed as maximal)
          While p2 < N:
            Check if number of maximal elements in the array is >= N/2
              If this is true, compare current number of turns with the best result \
                   and update it if necessary
              (additional matrix with number of maximal values between each pair of
                points may be used to count elements to the left of p1 and to the
                right of p2)
            Look at position [p1, p2] in the matrix. Mark it and if it contains \
                larger number of turns, update it
            Repeat:
              Increase p1 while it points to maximal value
              Increment p1 (to skip one non-maximum value)
              Increase p2 while it is possible to keep median at maximum value
            while median is not at maximum value
    

    To keep algorithm simple, I didn't mention special cases when group starts at position 0 or ends at position N, skipped initialization and didn't make any optimizations.

提交回复
热议问题