Leetcode上的一道算法题(Median of Two Sorted Arrays)

原题链接


描述:

There are two sorted arrays nums1 and nums2 of size m and n respectively.

Find the median of the two sorted arrays. The overall run time complexity should be O(log (m+n)).

Example 1:

nums1 = [1, 3]
nums2 = [2]

The median is 2.0

Example 2:

nums1 = [1, 2]
nums2 = [3, 4]

The median is (2 + 3)/2 = 2.5

翻译过来就是,给定两个已经升序排序过的数组,求这两个数组的中位数;中位数的定义为把两个数组合并过后进行升序排序后,处于数组中间的那个数,此时如果合并后的数组元素个数为偶数,则为中间两个数的平均值。

初看起来这就是个寻找第k小数的问题,解决方案有很多,最简单的就是采用归并排序的思想把两个数组进行合并,然后取中间的数就可以了。但问题在于,这个题目限定了时间复杂度为O(log(m+n)),而合并算法的时间复杂度为O(nlogn),显然不合题意。另外一个方法是设置一个双指针,一开始都指向两个数组的开头,不停地比较两个指针指向的元素的大小,指向小元素的指针的往前移一个元素去追指向大元素的指针,一直移动(len1+len2)/2次后就能得到中位数,但是这个算法的时间复杂度仍然不符合题意,为O(n)

但是注意到这个题目给定的数组已经是排过序的了,算法导论中对order statistic问题进行过讨论,因此,在有序又要求log级的时间复杂度,可以考虑分治策略,采用二分法。

大方向定好了,但是并不清楚具体要怎么去完成这个二分法,我们应该对什么去做二分?其实这个题目需要找的就是第k小的元素问题,假设我们的第k小的数是在第一个数组中找了p次,然后在第二个数组中找了q次,那么满足关系:p+q=k。进一步的,寻找第k小的数的过程就是寻找p和q的过程,k我们是知道的,但是p和q是不知道的,因此事实上我们的目标就是去搜索p(找到了p就等于找到了q),因此我们二分法的目标,事实上就是二分k来找p。

我们先定义以下形式的函数用来寻找第k小的数:

findKth(nums1, nums2, start1, len1, start2, len2, k)

nums1和nums2表示原始的两个数组,start1、len1表示nums1数组中以start1位置开始、len1长度的一个子数组;start2、lens2表示nums2数组中以start2位置开始、len2长度的一个子数组;k表示从这两个子数组中找到第k小的数。之所以提供start1、len1、start2、len2,是因为经验告诉我们分治法解决问题都是递归的,我们在二分的时候就需要记录这些相关的数据。

我们的外层入口就应该是这样子的:

if(len1+len2是偶数) {
  return (findKth(nums1, nums2, 0, len1, 0, len2, (len1+len2)/2) + 
        findKth(nums1, nums2, 0, len1, 0, len2, (len1+len2)/2 + 1)) / 2;
}
else {
  return findKth(nums1, nums2, 0, len1, 0, len2 (len1+len2)/2);
}

下面就是具体对于findKth的实现了。

首先,我们知道,我们需要对k进行二分:

findKth(nums1, nums2, start1, len1, start2, len2, k) {
  p = k / 2;
}

这个p怎么用呢?假设我们在nums1中取nums1[start1+p-1],就表示我们在nums1中“前进”了p个元素,且这p个元素是有序的,相应的,q = k - p,我们需要在nums2中前进q个元素:

findKth(nums1, nums2, start1, len1, start2, len2, k) {
  p = k / 2;
  q = k - p;
}

假如这个时候nums1[start1 + p - 1]等于nums2[start + q - 1],这说明kth = nums1[start1 + p - 1] = nums2[start + q - 1] = 第k小的数,为什么呢?这里要注意nums1和nums2都是有序的,因此它们的子数组也是有序的;假设把两个子数组数组合并,那么在nums1子数组中排在kth前面的数在合并后的数组中一定还是排在kth前面,同理在nums2子数组中排在kth前面的数在合并后的数组中也一定还是排在kth前面,它们的具体顺序我们不关心也不必关心,我们只需要知道这样一来在合并后的数组中就有p-1+q-1=k-2个数在kth前面,因此kth一定就是第k小的那个数

如果nums1[start1 + p - 1]大于nums2[start + q - 1],这里就出现了一个需要注意的情况,这意味着nums2子数组中的前q个数一定都是小于nums1[start1 + p - 1]的(再次注意,nums1和nums2都是有序的),而q<k,这也就意味着第k小的数一定不会出现在nums2的子数组的前q个数中。这启发我们在这个时候就可以抛弃掉前q个数,重新用一个新的子数组进行搜索,注意,进一步的搜索中由于抛弃掉了q(p)个数,因此下一步在子数组中的搜索中,事实上就是在搜索第k-q(k-p)小的元素了:

findKth(nums1, nums2, start1, len1, start2, len2, k) {
  p = k / 2;
  q = k - p;
  if(nums1[start1 + p - 1] == nums2[start2 + q - 1]) {
    return nums1[start1 + p - 1];
  }
  else if(nums1[start1 + p - 1] > nums2[start2 + q - 1]) {
    return findKth(nums1, nums2, start1, len1, start2 + q, len2 - q, k - q);
  }
  else if(nums1[start1 + p - 1] < nums2[start2 + q - 1]) {
    return findKth(nums1,  nums2, start1 + p, len1 - p, start2, len2, k - p);
  }
}

上面的框架大致已经描述清楚了我们的二分搜索算法,下一步就需要考虑退出条件。

退出条件有这么一些:

  1. 在某一步搜索中子数组的长度为0了,这表示有一个数组中的元素完全被抛弃掉,此时另外一个子数组的第k个元素就是我们要求的第k小的元素;
  2. 在不满足1的情况下,出现k=1的情况,这表示需要在两个子数组中找第1小的元素,此时简单地比较一下两个子数组的第一个元素就行了;
  3. nums1[start1 + p - 1] == nums2[start2 + q - 1]

因此可以进一步写成:

findKth(nums1, nums2, start1, len1, start2, len2, k) {
  if(某个子数组的长度为零) {
    return 另外一个子数组的第k个元素;
  }
  
  if(k == 1) {
    return min(nums1[start1], nums2[start2]);
  }
    
  p = k / 2;
  q = k - p;
  if(nums1[start1 + p - 1] == nums2[start2 + q - 1]) {
    return nums1[start1 + p - 1];
  }
  else if(nums1[start1 + p - 1] > nums2[start2 + q - 1]) {
    return findKth(nums1, nums2, start1, len1, start2 + q, len2 - q, k - q);
  }
  else if(nums1[start1 + p - 1] < nums2[start2 + q - 1]) {
    return findKth(nums1, nums2, start1 + p, len1 - p, start2, len2, k - p);
  }
}

为了方便考虑问题,不失一般性,我们要求nums1永远是那个长度较短的数组:

findKth(nums1, nums2, start1, len1, start2, len2, k) {
  if(len1 > len2) {
    return findKth(nums2, nums1, start2, len2, start2, start1);
  }
  
  if(len1 == 0) {
    return nums2[start2 + k - 1];
  }
  
  if(k == 1) {
    return min(nums1[start1], nums2[start2]);
  }
    
  p = k / 2;
  q = k - p;
  if(nums1[start1 + p - 1] == nums2[start2 + q - 1]) {
    return nums1[start1 + p - 1];
  }
  else if(nums1[start1 + p - 1] > nums2[start2 + q - 1]) {
    return findKth(nums1,  nums2, start1, len1, start2 + q, len2 - q, k - q);
  }
  else if(nums1[start1 + p - 1] < nums2[start2 + q - 1]) {
    return findKth(nums1,  nums2, start1 + p, len1 - p, start2, len2, k - p);
  }
}

此外还有一个容易被忽略的边界问题,那就是p=k/2这一句,如果p大于len1的话,就会出现越界访问的问题,这个时候需要对其进行控制:

findKth(nums1, nums2, start1, len1, start2, len2, k) {
  if(len1 > len2) {
    return findKth(nums2, nums1, start2, len2, start2, start1);
  }
  
  if(len1 == 0) {
    return nums2[start2 + k - 1];
  }
  
  if(k == 1) {
    return min(nums1[start1], nums2[start2]);
  }
    
  p = min(k / 2, len1);
  q = k - p;
  if(nums1[start1 + p - 1] == nums2[start2 + q - 1]) {
    return nums1[start1 + p - 1];
  }
  else if(nums1[start1 + p - 1] > nums2[start2 + q - 1]) {
    return findKth(nums1, start1, len1, start2 + q, len2 - q, k - q);
  }
  else if(nums1[start1 + p - 1] > nums2[start2 + q - 1]) {
    return findKth(nums1, start1 + p, len1 - p, start2, len2, k - p);
  }
}

分析到这里,基本上这个问题就解决了,不过需要说的是,对于p=min(k/2,len1)这一句,这里看起来应该就是二分法的比较关键的一个地方了,事实上我们把2换成3、4、5、6……都是可以的,因为二分法搜索事实上就是个碰运气的过程,不过需要注意的是,这里p不能为0,否则在nums1中等于是没有做“前进”的动作,这是不允许的,因此更加健壮的描述应该为:

p = min(max(k/2, 1), len1);

即二分过程中,每一次迭代至少要在nums1中“前进”一步。

整个程序的C++代码如下:

#include <vector>

class Solution {
private:
    double findKth(vector<int>& nums1, vector<int>& nums2, int start1, int len1, int start2, int len2, int k) {
        if (len1 > len2) {
            return findKth(nums2, nums1, start2, len2, start1, len1, k);
        }

        if (len1 == 0) {
            return nums2[start2 + k - 1];
        }

        if (k == 1) {
            return min(nums1[start1], nums2[start2]);
        }

        int p1 = min(k / 2, len1);
        int p2 = k - p1;
        if (nums1[start1 + p1 - 1] > nums2[start2 + p2 - 1]) {
            return findKth(nums1, nums2, start1, len1, start2 + p2, len2 - p2, k - p2);
        }
        else if(nums1[start1 + p1 - 1] < nums2[start2 + p2 - 1]){
            return findKth(nums1, nums2, start1 + p1, len1 - p1, start2, len2, k - p1);
        }
        else {
            return nums1[start1 + p1 - 1];
        }

    }

public:
    double findMedianSortedArrays(vector<int>& nums1, vector<int>& nums2) {
        int len = nums1.size() + nums2.size();

        if (!(len & 0x01)) {
            return (findKth(nums1, nums2, 0, nums1.size(), 0, nums2.size(), len / 2)
                + findKth(nums1, nums2, 0, nums1.size(), 0, nums2.size(), len / 2 + 1)
                ) / 2.0f;
        }
        else {
            return findKth(nums1, nums2, 0, nums1.size(), 0, nums2.size(), len / 2 + 1);
        }
    }
};

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,332评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,508评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,812评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,607评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,728评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,919评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,071评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,802评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,256评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,576评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,712评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,389评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,032评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,798评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,026评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,473评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,606评论 2 350

推荐阅读更多精彩内容