import static java.
lang.
Math.
max; import static java.
lang.
Math.
min;
class KthLargestTwoArrays {
/**
* Given two sorted arrays of equal sizes, finds the median of the
* array that comes out after `a` and `b` being merged. Runs in
* linear time and does not require any extra space.
*
* The algorithm is similar to the arrays merging, but it
* does not save the merged array, it just iterates through both
* of them by n+1 items (thus stopping in the middle of an imaginary
* merged array) and memorizes two consecutive elements. These
* are the two elements laying exactly in the middle of merged array.
*
* (Since the length of the merged array is even, its median
* is an average value of two elements in the middle.)
*
* @param a a sorted array
* @param b a sorted array
*/
static double medianTwoArraysLinear(double[] a, double[] b) {
if (a.length != b.length) {
"Expected: len(a) == len(b)."
+ " Got: len(a) = %d, len(b) = %d",
a.length, b.length
));
}
final int n = a.length;
int i = 0, j = 0;
// two consecutive elements of the merged array
for (int k = 0; k < n + 1; k++) {
// here we also handle a special case when every element
// of the first array is smaller than the smallest element
// of the second array (and vice versa)
if (j == n || i < n && a[i] < b[j]) {
// put the previous value into the buffer
mid1 = mid2;
mid2 = a[i];
i++;
} else {
mid1 = mid2;
mid2 = b[j];
j++;
}
}
return (mid1 + mid2) / 2;
}
/**
* A logarithmic-time algorithm that solves the same problem.
* The basic idea is to compute the medians of both of the
* arrays (well, these are not exactly "medians", in case the
* array length is even, mid = A[n / 2], not the average value):
* A = [a1, a2, ..., mid1, ..., an]
* B = [b1, b2, ..., mid2, ..., bn]
* and after that there are three cases to consider:
*
* - mid1 == mid2, which means (given len(A) == len(B)) that
* in every array there are n/2 elemets on the left side
* from the mid, so in the merged array mid1 and mid2 would
* be on n/2 + n/2 = n and n+1 positions (thus mid1 is the
* median of the merged array)
*
* - mid1 < mid2 means that the greatest element of the left
* side of the array A (A_left) is at most as large as the
* greatest element of B_left. And similarly the smallest
* element of B_right is at least as large as the smallest
* element of A_right.
*
* This means that the left side of the imaginary merged array
* for sure would not end with an element from A_left, and
* the right side will not start with an value from B_right.
* (It can be easily seen from the example down below, just
* follow the steps of the merge algorithm.)
*
* Thus the desired median elements are somewhere in
* A_right or B_left. When computing the median we can get rid
* of the same amount of elements on the right as on the left side
* and the result will not change in case the elements on the left
* are smaller than the median, meanwhile elements on the right
* are greater. So we can do here: A_left and B_right can be
* thrown away leaving us with just A_right and B_left -
* now we have the same problem but of half of the size.
*
* Example:
* a = [ 1, 2, 3, 4, 8 ]
* b = [ 0, 3, 4, 5, 6 ]
* Merged: [ 0, 1*, 2*, 3, 3* | 4, 4*, 5, 6, 8* ]
* (elements from the first array are marked with asterisks)
*
* - mid1 > mid2 case is handled similarly. */
static double medianTwoArrays(double[] a, double[] b) {
if (a.length != b.length) {
"Expected: len(a) == len(b)."
+ "Got: len(a) = %d, len(b) = %d",
a.length, b.length
));
}
final int n = a.length;
int lo1 = 0, hi1 = n; // left subarray
int lo2 = 0, hi2 = n; // right subarray
while (hi1 - lo1 > 2) {
double mid1 = a[(lo1 + hi1) >>> 1];
double mid2 = b[(lo2 + hi2) >>> 1];
if (mid1 > mid2) {
double[] arraySwap = a;
a = b;
b = arraySwap;
int swap = lo1;
lo1 = lo2;
lo2 = swap;
swap = hi1;
hi1 = hi2;
hi2 = swap;
} else if (mid1 == mid2) {
return mid1;
}
lo1 = (lo1 + hi1) >>> 1;
hi2 = (lo2 + hi2 + 1) >>> 1;
}
if (hi1 - lo1 == 1) {
return (a[lo1] + b[lo2]) / 2;
} else {
assert hi1 - lo1 == 2;
// [a0, a1] [b0, b1] case
// possible merged arrays:
// a0 < b0
// [a0, a1, b0, b1] mid = avg(a1, b0) -- a1 < b0 < b1
// [a0, b0, a1, b1] mid = avg(a1, b0) -- a1 < b1
// [a0, b0, b1, a1] mid = avg(b1, b0) -- a1 > b1
//
// a0 > b0
// [b0, a0, b1, a1] mid = avg(a0, b1) -- a1 > b1
// [b0, a0, a1, b1] mid = avg(a0, a1) -- a1 < b1
// [b0, b1, a0, a1] mid = avg(a0, b1) -- a1 > a0 > b1
return (max(a[lo1], b[lo2])
+ min(a[lo1 + 1], b[lo2 + 1])) / 2;
}
}
public static void main
(String[] args
) { double[] a = { 1, 12, 15, 26, 38 };
double[] b = { 2, 13, 17, 30, 45 };
System.
out.
println(medianTwoArrays
(a, b
)); System.
out.
println(medianTwoArraysLinear
(a, b
)); }
}
import static java.lang.Math.max;
import static java.lang.Math.min;

class KthLargestTwoArrays {
    /**
     * Given two sorted arrays of equal sizes, finds the median of the
     * array that comes out after `a` and `b` being merged. Runs in
     * linear time and does not require any extra space.
     *
     * The algorithm is similar to the arrays merging, but it
     * does not save the merged array, it just iterates through both
     * of them by n+1 items (thus stopping in the middle of an imaginary
     * merged array) and memorizes two consecutive elements. These
     * are the two elements laying exactly in the middle of merged array.
     *
     * (Since the length of the merged array is even, its median
     * is an average value of two elements in the middle.)
     *
     * @param a a sorted array
     * @param b a sorted array
     */
    static double medianTwoArraysLinear(double[] a, double[] b) {
        if (a.length != b.length) {
            throw new IllegalArgumentException(String.format(
                "Expected: len(a) == len(b)."
                + " Got: len(a) = %d, len(b) = %d",
                a.length, b.length
            ));
        }
        
        final int n = a.length;
        int i = 0, j = 0;
        // two consecutive elements of the merged array
        double mid1 = Double.NaN, mid2 = Double.NaN;
        for (int k = 0; k < n + 1; k++) {
            // here we also handle a special case when every element
            // of the first array is smaller than the smallest element
            // of the second array (and vice versa)
            if (j == n || i < n && a[i] < b[j]) {
                // put the previous value into the buffer
                mid1 = mid2;
                mid2 = a[i];
                i++;
            } else {
                mid1 = mid2;
                mid2 = b[j];
                j++;
            }
        }
        return (mid1 + mid2) / 2;
    }
    
    /**
     * A logarithmic-time algorithm that solves the same problem.
     * The basic idea is to compute the medians of both of the
     * arrays (well, these are not exactly "medians", in case the
     * array length is even, mid = A[n / 2], not the average value):
     *     A = [a1, a2, ..., mid1, ..., an] 
     *     B = [b1, b2, ..., mid2, ..., bn]
     * and after that there are three cases to consider:
     *
     *  - mid1 == mid2, which means (given len(A) == len(B)) that
     *    in every array there are n/2 elemets on the left side
     *    from the mid, so in the merged array mid1 and mid2 would
     *    be on n/2 + n/2 = n and n+1 positions (thus mid1 is the
     *    median of the merged array)
     *
     *  - mid1 &lt; mid2 means that the greatest element of the left
     *    side of the array A (A_left) is at most as large as the
     *    greatest element of B_left. And similarly the smallest
     *    element of B_right is at least as large as the smallest
     *    element of A_right.
     *
     *    This means that the left side of the imaginary merged array
     *    for sure would not end with an element from A_left, and
     *    the right side will not start with an value from B_right.
     *    (It can be easily seen from the example down below, just
     *    follow the steps of the merge algorithm.)
     *
     *    Thus the desired median elements are somewhere in
     *    A_right or B_left. When computing the median we can get rid
     *    of the same amount of elements on the right as on the left side
     *    and the result will not change in case the elements on the left
     *    are smaller than the median, meanwhile elements on the right
     *    are greater. So we can do here: A_left and B_right can be
     *    thrown away leaving us with just A_right and B_left -
     *    now we have the same problem but of half of the size.
     *
     *    Example:
     *    a = [ 1, 2, 3, 4, 8 ]
     *    b = [ 0, 3, 4, 5, 6 ]
     *    Merged: [ 0, 1*, 2*, 3, 3* | 4, 4*, 5, 6, 8* ]
     *    (elements from the first array are marked with asterisks)
     *
     *  - mid1 &gt; mid2 case is handled similarly.  */ 
    static double medianTwoArrays(double[] a, double[] b) {
        if (a.length != b.length) {
            throw new IllegalArgumentException(String.format(
                "Expected: len(a) == len(b)."
                + "Got: len(a) = %d, len(b) = %d",
                a.length, b.length
            ));
        } 
        final int n = a.length;
        int lo1 = 0, hi1 = n; // left subarray
        int lo2 = 0, hi2 = n; // right subarray

        while (hi1 - lo1 > 2) {
            double mid1 = a[(lo1 + hi1) >>> 1];
            double mid2 = b[(lo2 + hi2) >>> 1];

            if (mid1 > mid2) {
                double[] arraySwap = a;
                a = b;
                b = arraySwap;

                int swap = lo1;
                lo1 = lo2;
                lo2 = swap;

                swap = hi1;
                hi1 = hi2;
                hi2 = swap;
            } else if (mid1 == mid2) {
                return mid1;
            }
            lo1 = (lo1 + hi1) >>> 1;
            hi2 = (lo2 + hi2 + 1) >>> 1;
        }

        if (hi1 - lo1 == 1) {
            return (a[lo1] + b[lo2]) / 2;
        } else {
            assert hi1 - lo1 == 2;
            // [a0, a1] [b0, b1] case
            // possible merged arrays:
            //   a0 < b0
            //   [a0, a1, b0, b1] mid = avg(a1, b0) -- a1 < b0 < b1
            //   [a0, b0, a1, b1] mid = avg(a1, b0) -- a1 < b1
            //   [a0, b0, b1, a1] mid = avg(b1, b0) -- a1 > b1
            //   
            //   a0 > b0
            //   [b0, a0, b1, a1] mid = avg(a0, b1) -- a1 > b1
            //   [b0, a0, a1, b1] mid = avg(a0, a1) -- a1 < b1
            //   [b0, b1, a0, a1] mid = avg(a0, b1) -- a1 > a0 > b1
            return (max(a[lo1], b[lo2])
                    + min(a[lo1 + 1], b[lo2 + 1])) / 2;
        }
    }

    public static void main(String[] args) {
        double[] a = { 1, 12, 15, 26, 38 };
        double[] b = { 2, 13, 17, 30, 45 };
        System.out.println(medianTwoArrays(a, b));
        System.out.println(medianTwoArraysLinear(a, b));
    }
}