LeetCode No4 寻找两个正序数组的中位数

题目

给定两个大小分别为 m 和 n 的正序(从小到大)数组 nums1 和 nums2。请你找出并返回这两个正序数组的 中位数 。

算法的时间复杂度应该为 O(log (m+n)) 。

示例 1:

输入:nums1 = [1,3], nums2 = [2]
输出:2.00000
解释:合并数组 = [1,2,3] ,中位数 2

示例 2:

输入:nums1 = [1,2], nums2 = [3,4]
输出:2.50000
解释:合并数组 = [1,2,3,4] ,中位数 (2 + 3) / 2 = 2.5

提示:

nums1.length == m
nums2.length == n
0 <= m <= 1000
0 <= n <= 1000
1 <= m + n <= 2000
-10^6 <= nums1[i], nums2[i] <= 10^6

思路

暴力解法

如果不考虑时间复杂度的要求,这道题可以用暴力去解决,根据中位数的定义,中位数一定是有序序列的中间一位或者两位的平均值,那么我只要同时开始遍历两个数组,找到中间的两个数,然后根据两个数组长度总和的奇偶,奇数的话就取小值,偶数就取两者的平均数。

二分查找

暴力的解法固然可以AC,但是还是不满足题目中对于时间复杂度的要求,而且对于这种在有序的数组中找数的题目,第一反应都是用二分去做,只是这题比较特殊,是要在两个数组里面求合并后的值,当然我们可以合并数组后再二分查找,但事实上,既然暴力解法我们都不需要真正的合并数组,那使用二分查找的时候肯定也可以不用。
于是我们再看下中位数的定义:

中位数(Median)又称中值,统计学中的专有名词,是按顺序排列的一组数据中居于中间位置的数,代表一个样本、种群或概率分布中的一个数值,其可将数值集合划分为相等的上下两部分。

既然中位数可以将数值集合划分为相等的两部分,那么我们先在一个数组里面看中位数,将数组从中间分割开来,中位数一定在坐标i和i-1两个数中,如果是数组长度为奇数那么取min(nums[i-1], nums[i]),而如果是偶数则为 (nums[i-1]+nums[i])/2。

    1 3 5 | 8(i) 9

    2 4 6 | 7(i) 10 12

同样如果是两个数组,将两个数组从中间分隔开,使得左边的数据个数=右边的数据个数或者左边比右边多一个,那这个时候基本上就能确定中位数的位置了

    1 3 5 | 8(i) 9
    2 4 6 | 7(j) 10 12

当然仅仅只是数量上满足条件当然还不够,如果中位数就在分割线附近的话,那么分割线附近的4个数,在合并两个数组后应该是一个连续字串,也就是需要满足
nums1[i-1]<=nums2[j] && nums1[j-1]<=nums2[i]
这个条件,于是我们二分查找的条件也就出来了,但是还是需要注意如下四种边界情况。

1 2 3(i) |
| 4(j) 5 6 7
| 4(i) 5 6
1 2 3 4(j) |
1 2 3(i) |
| 4(j) 5 6 
| 4(i) 5 6 
1 2 3(j) |

AC代码

暴力解法

点击查看代码
class Solution {
    public double findMedianSortedArrays(int[] nums1, int[] nums2) {
        int n = nums1.length;
        int m = nums2.length;
        boolean flag = (n+m)%2==0;
        int mid = (n+m)/2+1;
        int i = 0, j = 0;
        int a = 0, b = 0;
        int cnt = 1;
        while( i<n || j < m ) {
            a = b;
            if( i!=n ) {
                if( j!=m ) {
                    if( nums1[i] < nums2[j] ) {
                        b = nums1[i];
                        i ++;
                    } else {
                        b = nums2[j];
                        j ++;
                    }
                } else {
                    b = nums1[i];
                    i ++;
                }
            } else {
                b = nums2[j];
                j ++;
            }
            if( cnt == mid ) {
                break;
            }
            cnt ++;
        }
        if( flag ) {
            return (a+b)/2.0;
        }
        return b;
    }
}

二分查找

点击查看代码
class Solution {
    public double findMedianSortedArrays(int[] nums1, int[] nums2) {
        int n = nums1.length;
        int m = nums2.length;
        if( n > m ) {
            return findMedianSortedArrays(nums2, nums1);
        }
        int left = 0;
        int rigth = n;
        int i = 0;
        int j = 0;

        // 定义的分割线需要满足 nums1[i-1]<=nums2[j] && nums1[j-1]<=nums2[i]
        // 1 3 5 | 8(i) 9
        // 2 4 6 | 7(j) 10 12
        // 左边数字个数应该为 (n+m+1)/2 = i + j
        while( left<rigth ) {
            i = (left + rigth +1 ) / 2;
            j = (n+m+1)/2-i;
            // 二分确定能满足条件的分割线
            if( nums1[i-1]>nums2[j] ) {
                rigth = i - 1;
            } else {
                left = i;
            }
        }
        i = left;
        j = (n+m+1)/2-i;
        // 边界处理
        int numsI1 = i==0? Integer.MIN_VALUE: nums1[i-1];
        int numsI = i==n? Integer.MAX_VALUE: nums1[i];
        int numsJ1 = j==0? Integer.MIN_VALUE: nums2[j-1];
        int numsJ = j==m? Integer.MAX_VALUE: nums2[j];
        if( (n+m)%2 == 1 ) {
            return Math.max(numsI1, numsJ1);
        } else {
            return (Math.max(numsI1, numsJ1) + Math.min(numsI, numsJ)) * 0.5;
        }
    }
}
posted @ 2022-04-09 00:23  Asimple  阅读(25)  评论(0编辑  收藏  举报