目录

0004:寻找两个正序数组的中位数(★★)

力扣第 4 题

题目

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

算法的时间复杂度应该为 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
  • -106 <= nums1[i], nums2[i] <= 106

相似问题:

分析

  • 假设中位数为 x,那么可以用 x 将 nums1 和 nums2 划分为两部分,满足
    • nums1[i-1]<=x<=nums1[i]
    • nums2[j-1]<=x<=nums2[j]
    • i+j=(m+n)//2
  • 反过来,如果找到一组 (i,j),满足
    • nums1[i-1]<=nums2[j]
    • num2[j-1]<=nums1[i]
    • i+j=(m+n)//2 便定位了中位数
  • 注意到 nums1 和 nums2 都递增,那么只需找到第一个 i 满足
    • nums2[(m+n)//2-i-1]<=nums1[i]
  • 这就是典型的二分查找了

再考虑边界情况:

  • 为了方便,当 m>n 时,交换 nums1、nums2,不需计算 i 的边界
  • m+n 为偶数时,要找到两个中位数取平均
  • 为了方便处理 i 或 j 在最左/右的情况,在 nums1/nums2 前后添加哨兵

解答

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
class Solution:
    def findMedianSortedArrays(self, nums1: List[int], nums2: List[int]) -> float:
        A,B = [-inf]+nums1+[inf],[-inf]+nums2+[inf]
        if len(A)>len(B):
            A,B = B,A
        m,n = len(A),len(B)
        k = (m+n)//2
        i = bisect_left(range(m),True,key=lambda i:A[i]>=B[k-i-1])
        a = max(A[i-1],B[k-i-1])
        b = min(A[i],B[k-i])
        return b if (m+n)%2 else (a+b)/2

时间 $O(log \ min(M,N))$,0 ms