Find the first element in a sorted array that is greater than the target

后端 未结 8 531
青春惊慌失措
青春惊慌失措 2020-11-30 17:40

In a general binary search, we are looking for a value which appears in the array. Sometimes, however, we need to find the first element which is either greater or less than

相关标签:
8条回答
  • 2020-11-30 18:01

    kind =0: exact match, kind=1: just grater than x, kind=-1: just smaller than x;

    It returns -1 if no match is found.

    #include <iostream>
    #include <algorithm>
    
    using namespace std;
    
    
    int g(int arr[], int l , int r, int x, int kind){
        switch(kind){
        case 0: // for exact match
            if(arr[l] == x) return l;
            else if(arr[r] == x) return r;
            else return -1;
            break;
        case 1: // for just greater than x
            if(arr[l]>=x) return l;
            else if(arr[r]>=x) return r;
            else return -1;
            break;
        case -1: // for just smaller than x
            if(arr[r]<=x) return r;
            else if(arr[l] <= x) return l;
            else return -1;
            break;
        default:
            cout <<"please give "kind" as 0, -1, 1 only" << ednl;
        }
    }
    
    int f(int arr[], int n, int l, int r, int x, int kind){
        if(l==r) return l;
        if(l>r) return -1;
        int m = l+(r-l)/2;
        while(m>l){
            if(arr[m] == x) return m;
            if(arr[m] > x) r = m;
            if(arr[m] < x) l = m;
            m = l+(r-l)/2;
        }
        int pos = g(arr, l, r, x, kind);
        return pos;
    }
    
    int main()
    {
        int arr[] = {1,2,3,5,8,14, 22, 44, 55};
        int n = sizeof(arr)/sizeof(arr[0]);
        sort(arr, arr+n);
        int tcs;
        cin >> tcs;
        while(tcs--){
            int l = 0, r = n-1, x = 88, kind = -1; // you can modify these values
            cin >> x;
            int pos = f(arr, n, l, r, x, kind);
            // kind =0: exact match, kind=1: just grater than x, kind=-1: just smaller than x;
            cout <<"position"<< pos << " Value ";
            if(pos >= 0) cout << arr[pos];
            cout << endl;
        }
        return 0;
    }
    
    0 讨论(0)
  • 2020-11-30 18:07

    One way of thinking about this problem is to think about doing a binary search over a transformed version of the array, where the array has been modified by applying the function

    f(x) = 1 if x > target
           0 else
    

    Now, the goal is to find the very first place that this function takes on the value 1. We can do that using a binary search as follows:

    int low = 0, high = numElems; // numElems is the size of the array i.e arr.size() 
    while (low != high) {
        int mid = (low + high) / 2; // Or a fancy way to avoid int overflow
        if (arr[mid] <= target) {
            /* This index, and everything below it, must not be the first element
             * greater than what we're looking for because this element is no greater
             * than the element.
             */
            low = mid + 1;
        }
        else {
            /* This element is at least as large as the element, so anything after it can't
             * be the first element that's at least as large.
             */
            high = mid;
        }
    }
    /* Now, low and high both point to the element in question. */
    

    To see that this algorithm is correct, consider each comparison being made. If we find an element that's no greater than the target element, then it and everything below it can't possibly match, so there's no need to search that region. We can recursively search the right half. If we find an element that is larger than the element in question, then anything after it must also be larger, so they can't be the first element that's bigger and so we don't need to search them. The middle element is thus the last possible place it could be.

    Note that on each iteration we drop off at least half the remaining elements from consideration. If the top branch executes, then the elements in the range [low, (low + high) / 2] are all discarded, causing us to lose floor((low + high) / 2) - low + 1 >= (low + high) / 2 - low = (high - low) / 2 elements.

    If the bottom branch executes, then the elements in the range [(low + high) / 2 + 1, high] are all discarded. This loses us high - floor(low + high) / 2 + 1 >= high - (low + high) / 2 = (high - low) / 2 elements.

    Consequently, we'll end up finding the first element greater than the target in O(lg n) iterations of this process.

    EDIT: Here's a trace of the algorithm running on the array 0 0 1 1 1 1.

    Initially, we have

    0 0 1 1 1 1
    L = 0       H = 6
    

    So we compute mid = (0 + 6) / 2 = 3, so we inspect the element at position 3, which has value 1. Since 1 > 0, we set high = mid = 3. We now have

    0 0 1
    L     H
    

    We compute mid = (0 + 3) / 2 = 1, so we inspect element 1. Since this has value 0 <= 0, we set mid = low + 1 = 2. We're now left with L = 2 and H = 3:

    0 0 1
        L H
    

    Now, we compute mid = (2 + 3) / 2 = 2. The element at index 2 is 1, and since 1 ≥ 0, we set H = mid = 2, at which point we stop, and indeed we're looking at the first element greater than 0.

    Hope this helps!

    0 讨论(0)
  • 2020-11-30 18:07

    How about the following recursive approach:

        public static int minElementGreaterThanOrEqualToKey(int A[], int key,
            int imin, int imax) {
    
        // Return -1 if the maximum value is less than the minimum or if the key
        // is great than the maximum
        if (imax < imin || key > A[imax])
            return -1;
    
        // Return the first element of the array if that element is greater than
        // or equal to the key.
        if (key < A[imin])
            return imin;
    
        // When the minimum and maximum values become equal, we have located the element. 
        if (imax == imin)
            return imax;
    
        else {
            // calculate midpoint to cut set in half, avoiding integer overflow
            int imid = imin + ((imax - imin) / 2);
    
            // if key is in upper subset, then recursively search in that subset
            if (A[imid] < key)
                return minElementGreaterThanOrEqualToKey(A, key, imid + 1, imax);
    
            // if key is in lower subset, then recursively search in that subset
            else
                return minElementGreaterThanOrEqualToKey(A, key, imin, imid);
        }
    }
    
    0 讨论(0)
  • 2020-11-30 18:09

    You can use std::upper_bound if the array is sorted (assuming n is the size of array a[]):

    int* p = std::upper_bound( a, a + n, x );
    if( p == a + n )
         std::cout << "No element greater";
    else
         std::cout << "The first element greater is " << *p
                   << " at position " << p - a;
    
    0 讨论(0)
  • 2020-11-30 18:09

    My following implementation uses condition bottom <= top which is different from the answer of @templatetypedef.

    int FirstElementGreaterThan(int n, const vector<int>& values) {
      int B = 0, T = values.size() - 1, M = 0;
      while (B <= T) { // B strictly increases, T strictly decreases
        M = B + (T - B) / 2;
        if (values[M] <= n) { // all values at or before M are not the target
          B = M + 1;
        } else {
          T = M - 1;// search for other elements before M
        }
      }
      return T + 1;
    }
    
    0 讨论(0)
  • 2020-11-30 18:15
    public static int search(int target, int[] arr) {
            if (arr == null || arr.length == 0)
                return -1;
            int lower = 0, higher = arr.length - 1, last = -1;
            while (lower <= higher) {
                int mid = lower + (higher - lower) / 2;
                if (target == arr[mid]) {
                    last = mid;
                    lower = mid + 1;
                } else if (target < arr[mid]) {
                    higher = mid - 1;
                } else {
                    lower = mid + 1;
           }
        }
        return (last > -1 && last < arr.length - 1) ? last + 1 : -1;
    }
    

    If we find 'target == arr[mid]', then any previous element would be either less than or equal to target. Hence, lower boundary is set as 'lower=mid+1'. Also, 'last' is the last index of 'target'. Finally we return 'last+1' - taking care of boundary conditions.

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