longest nondecreasing subsequence in O(nlgn)

前端 未结 5 1913
无人及你
无人及你 2021-01-06 04:07

I have the following algorithm which works well

I tried explaining it here for myself http://nemo.la/?p=943 and it is explained here http://www.geeksforgeeks.org/lon

相关标签:
5条回答
  • 2021-01-06 04:28

    Just apply the longest increasing sub-sequence algorithm to ordered pair (A[i], i), by using a lexicographic compare.

    0 讨论(0)
  • 2021-01-06 04:29

    My Java version:

      public static int longestNondecreasingSubsequenceLength(List<Integer> A) {
        int n = A.size();
        int dp[] = new int[n];
        int max = 0;
        for(int i = 0; i < n; i++) {
            int el = A.get(i);
            int idx = Arrays.binarySearch(dp, 0, max, el);
            if(idx < 0) {
                idx = -(idx + 1);
            }
            if(dp[idx] == el) { // duplicate found, let's find the last one 
                idx = Arrays.binarySearch(dp, 0, max, el + 1);
                if(idx < 0) {
                    idx = -(idx + 1);
                }
            }
            dp[idx] = el;
            if(idx == max) {
                max++;
            }
        }
        return max;
    }
    
    0 讨论(0)
  • 2021-01-06 04:37

    To find the longest non-strictly increasing subsequence, change these conditions:

    1. If A[i] is smallest among all end candidates of active lists, we will start new active list of length 1.
    2. If A[i] is largest among all end candidates of active lists, we will clone the largest active list, and extend it by A[i].
    3. If A[i] is in between, we will find a list with largest end element that is smaller than A[i]. Clone and extend this list by A[i]. We will discard all other lists of same length as that of this modified list.

    to:

    1. If A[i] is smaller than the smallest of all end candidates of active lists, we will start new active list of length 1.
    2. If A[i] is largest among all end candidates of active lists, we will clone the largest active list, and extend it by A[i].
    3. If A[i] is in between, we will find a list with largest end element that is smaller than or equal to A[i]. Clone and extend this list by A[i]. We will discard all other lists of same length as that of this modified list.

    The fourth step for your example sequence should be:

    10 is not less than 10 (the smallest element). We find the largest element that is smaller than or equal to 10 (that would be s[0]==10). Clone and extend this list by 10. Discard the existing list of length 2. The new s becomes {10 10}.

    0 讨论(0)
  • 2021-01-06 04:38

    Your code nearly works except a problem in your binary_search() function, this function should return the index of the first element that's greater than the target element(x) since you want the longest non-decreasing sequence. Modify it to this, it'll be OK.

    If you use c++, std::lower_bound() and std::upper_bound() will help you get rid of this confusing problem. By the way, the if statement"if (height[s[k]] >= height[i])" is superfluous.

    int binary_search(int first, int last, int x) {
    
        while(last > first)
        {
            int mid = first + (last - first) / 2;
            if(height[s[mid]] > x)
                last = mid;
            else
                first = mid + 1;
        }
    
        return first; /* or last */
    }
    
    0 讨论(0)
  • 2021-01-06 04:46

    A completely different solution to this problem is the following. Make a copy of the array and sort it. Then, compute the minimum nonzero difference between any two elements of the array (this will be the minimum nonzero difference between two adjacent array elements) and call it δ. This step takes time O(n log n).

    The key observation is that if you add 0 to element 0 of the original array, δ/n to the second element of the original array, 2δ/n to the third element of the array, etc., then any nondecreasing sequence in the original array becomes a strictly increasing sequence in the new array and vice-versa. Therefore, you can transform the array this way, then run a standard longest increasing subsequence solver, which runs in time O(n log n). The net result of this process is an O(n log n) algorithm for finding the longest nondecreasing subsequence.

    For example, consider 30, 20, 20, 10, 10, 10, 10. In this case δ = 10 and n = 7, so δ / n ≈ 1.42. The new array is then

    40, 21.42, 22.84, 14.28, 15.71, 17.14, 18.57
    

    Here, the LIS is 14.28, 15.71, 17.14, 18.57, which maps back to 10, 10, 10, 10 in the original array.

    Hope this helps!

    0 讨论(0)
提交回复
热议问题