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)).

中位数:如果总数是偶数,那么中位数=中间两个数的平均数;如果总数是奇数,那么中位数=中间那个数的值

思路: O(logx)的算法,就想到二分法。二分法结束的条件是任何一个array只剩下一个元素了。每次递归(二分),去除某个array的一半。另外注意,每次二分至少要去掉一个元素。

class Solution {
public:
double findMedianSortedArrays(vector<int>& nums1, vector<int>& nums2) {
int size1 = nums1.size();
int size2 = nums2.size();
int size = size1 + size2;
int m = size >> ; if(nums1.empty()){
if(size% == ) return nums2[m];
else return (double) (nums2[m-] + nums2[m])/;
}
if(nums2.empty()){
if(size% == ) return nums1[m];
else return (double) (nums1[m-] + nums1[m])/;
} if(size% == ) return findK(nums1, nums2, , size1-, , size2-, m);
else return (double)(findK(nums1, nums2, , size1-, , size2-, m-)+findK(nums1, nums2, , size1-, , size2-, m))/;
} int findK(vector<int>& nums1, vector<int>& nums2,int s1, int e1, int s2, int e2, int k){
if(s1 == e1){
if(k == ) return min(nums1[s1],nums2[s2]); //为了防止s2+k-1越界
if(nums1[s1] < nums2[s2+k-]) return nums2[s2+k-];
else if(s2+k- == e2) return nums1[s1]; //为了防止s2+k越界
else return min(nums1[s1], nums2[s2+k]);
}
if(s2 == e2){
if(k == ) return min(nums1[s1],nums2[s2]); //为了防止s1+k-1越界
if(nums2[s2] < nums1[s1+k]) return max(nums2[s2],nums1[s1+k-]);
else if(s1+k- == e1) return nums2[s2]; //为了防止s1+k越界
else return min(nums2[s2], nums1[s1+k]);
} int m1 = s1+((e1-s1)>>);
int m2 = s2+((e2-s2)>>); int halfLen = (e1 - s1 + e2 - s2 + ) >> ;
if(k > halfLen){ //k is in the second half
if(nums1[m1] < nums2[m2]){ //delete first half of num1
return findK(nums1, nums2, m1+, e1, s2, e2, k-(m1-s1+)); //+1是考虑到m1==s1的情况,注意每次二分至少要去除一个元素
}
else{ //delete fist half of num2
return findK(nums1, nums2, s1, e1, m2+, e2, k-(m2-s2+));//+1是考虑到m2==s2的情况,注意每次二分至少要去除一个元素
}
}
else{ //k is in the first half
if(nums1[m1] < nums2[m2]){ //delete second half of num2
return findK(nums1, nums2, s1, e1, s2, m2, k);
}
else{ //delete second half of num1
return findK(nums1, nums2, s1, m1, s2, e2, k);
}
}
}
};

改进:每次去掉(size1+size2)/2个元素。

假设数组A和B如下,k=6,那么令i=k/2, j =k-k/2

注意: j =k-k/2,保证i+j涵盖整个数组,否则测试用例 [1,2,6] [3,4,5,7,8],k=5的时候将会选取4和6,而不是4和5,原因是在某次比较A[1]和B[1](只有4个,k=5少了一个),选取了A的right subarray[2,6]和B的left subarray[3,4],漏选了Target元素5。如果使用j=k-k/2,那么将会比较A[1]和B[2](5个),从而也会保留B[2]。

B的前三个元素可去掉比较好理解,那么为什么A的后5个元素也能取掉呢?

=>即使B的后7个都比A的前三个大,B的前三个+A的前三个也已经满足k个要求,不需要A的后5个元素了。

class Solution {
public:
double findMedianSortedArrays(vector<int>& nums1, vector<int>& nums2) {
int len = nums1.size() + nums2.size(); //see out to check null array
if(nums1.empty()){
if(len% == ) return nums2[(len >> )];
else return (double) (nums2[(len >> )-] + nums2[(len >> )])/;
}
if(nums2.empty()){
if(len% == ) return nums1[(len >> )];
else return (double) (nums1[(len >> )-] + nums1[(len >> )])/;
} if( len% == ) return findK(nums1,,nums1.size()-, nums2, , nums2.size()-,(len >> ));
else {
int m1 = findK(nums1,,nums1.size()-, nums2, , nums2.size()-,(len >> )-);
int m2 = findK(nums1,,nums1.size()-, nums2, , nums2.size()-,(len >> ));
return (double) (m1+m2)/;
} } int findK(vector<int>& nums1,int s1,int e1, vector<int>& nums2, int s2, int e2, int k) {
//termination
if(k==) return min(nums1[s1],nums2[s2]);
if(s1==e1 && s2 == e2) {
return max(nums1[s1],nums2[s2]);
}
if(s1 == e1){ // only one element in nums1
if(s2+k > e2){ //prevent overflow
return max(nums1[s1],nums2[e2]);
}
else if(nums1[s1] < nums2[s2+k-]) return nums2[s2+k-];
else if(nums1[s1] < nums2[s2+k]) return nums1[s1];
else return nums2[s2+k];
}
else if(s2 == e2){// only one element in nums2
if(s1+k > e1){ //prevent overflow
return max(nums1[e1],nums2[s2]);
}
else if( nums2[s2] < nums1[s1+k-]) return nums1[s1+k-];
else if(nums2[s2] < nums1[s1+k]) return nums2[s2];
else return nums1[s1+k];
} //make the length of first half of m1 + first half of m2 = k
int m1 = s1 + (k >> );
if(m1 > e1){ //check overflow
m1 = e1;
}
int m2 = s2 + k - (m1-s1+);
if(m2 > e2){ //check overflow
m2 = e2;
m1 = s1 + k - (m2-s2+);
} if((m1 == s1 && m2 == e2) || (m1==e1 && m2==s2)){
//try to make k-1
if(nums1[s1] < nums2[s2]) return findK(nums1,s1+,e1,nums2,s2,e2,k-);
else return findK(nums1,s1,e1,nums2,s2+,e2,k-);
} //compare two arrays
if(nums1[m1]<nums2[m2]){ //remove first half of num1 and second half of num2
return findK(nums1,m1,e1,nums2,s2,m2,k-(m1-s1)); //need to guarantee ! (m1==s1 && m2==e2)
}
else{//remove first half of num2 and second half of num1
return findK(nums1,s1,m1,nums2,m2,e2,k-(m2-s2)); // need to guarantee ! (m1==e1 && m2==s2)
}
} };

 

4. Median of Two Sorted Arrays(Array; Divide-and-Conquer)的更多相关文章

  1. 【转载】两个排序数组的中位数 / 第K大元素(Median of Two Sorted Arrays)

    转自 http://blog.csdn.net/zxzxy1988/article/details/8587244 给定两个已经排序好的数组(可能为空),找到两者所有元素中第k大的元素.另外一种更加具 ...

  2. 【算法之美】求解两个有序数组的中位数 — leetcode 4. Median of Two Sorted Arrays

    一道非常经典的题目,Median of Two Sorted Arrays.(PS:leetcode 我已经做了 190 道,欢迎围观全部题解 https://github.com/hanzichi/ ...

  3. [LintCode] Median of Two Sorted Arrays 两个有序数组的中位数

    There are two sorted arrays A and B of size m and n respectively. Find the median of the two sorted ...

  4. LeetCode 4 Median of Two Sorted Arrays (两个数组的mid值)

    题目来源:https://leetcode.com/problems/median-of-two-sorted-arrays/ There are two sorted arrays nums1 an ...

  5. Kotlin实现LeetCode算法题之Median of Two Sorted Arrays

    题目Median of Two Sorted Arrays(难度Hard) 方案1,数组合并&排序调用Java方法 import java.util.* class Solution { fu ...

  6. 【LeetCode】4. Median of Two Sorted Arrays (2 solutions)

    Median of Two Sorted Arrays There are two sorted arrays A and B of size m and n respectively. Find t ...

  7. LeetCode: Median of Two Sorted Arrays 解题报告

    Median of Two Sorted Arrays There are two sorted arrays A and B of size m and n respectively. Find t ...

  8. 《LeetBook》leetcode题解(4): Median of Two Sorted Arrays[H]——两个有序数组中值问题

    我现在在做一个叫<leetbook>的免费开源书项目,力求提供最易懂的中文思路,目前把解题思路都同步更新到gitbook上了,需要的同学可以去看看 书的地址:https://hk029.g ...

  9. LeetCode 004 Median of Two Sorted Arrays

    题目描述:Median of Two Sorted Arrays There are two sorted arrays A and B of size m and n respectively. F ...

随机推荐

  1. C#调用Excel宏

    using System; using Excel = Microsoft.Office.Interop.Excel; namespace WindowsFormsApplication1 { /// ...

  2. 转 - ubuntu 安装node.js 与 npm

    原文链接为: https://blog.csdn.net/wangtaoking1/article/details/78005038 这篇文章介绍如何在ubuntu环境下安装node环境. 我使用的系 ...

  3. Ps操作技巧(快捷键大全)

    一.工具箱(多种工具共用一个快捷键的可同时按[Shift]加此快捷键选取) 矩形.椭圆选框工具 [M] 移动工具 [V] 套索.多边形套索.磁性套索 [L] 魔棒工具 [W] 裁剪工具 [C] 切片工 ...

  4. Web API 源码剖析之全局配置

    Web API 源码剖析之全局配置 Web API  均指Asp.net Web API .本节讲述的是基于Web API 系统在寄宿于IIS. 本节主要讲述Web API全局配置.它是如何优雅的实现 ...

  5. python函数入门

    知识内容: 1.函数的作用 2.函数的定义与调用 3.函数的返回值 4.函数的参数 5.局部变量与全局变量 6.作用域 一.函数的作用 1.复用代码 将可能重复执行的代码封装成函数,并在需要执行的地方 ...

  6. VC如何得到一个文件夹的路径

    VC中没有现成的函数来选择一个文件夹,但这是经常会用到的,怎么办?自动动手,丰衣足食! 使用SHBrowseForFolder,代码如下: #include   int SelFolder(HWND ...

  7. tomcat启动原理

    2018年04月12日 19:55:22 太极小帅帅 阅读数:282   前言 一直在用Tomcat,但是对其启动原理一直没去研究,这里准备去面试,可能会问道.于是总结了下启动原理.完全凭感觉去揣测, ...

  8. GD库简介和使用

    简介 php并不仅限于创建html输出,它也可以创建和处理包括GIF,PNG,jpef,wbmp以及xpm在内的多种格式的图像.更加方便的是,php可以直接将图像数据库输出到浏览器.要想在php中使用 ...

  9. jsp 传多个值给后端

     页面上是这样 http://localhost:8080/smartcloset/getClothByCategory/1/11 直接用/分 后台是这样取的 @RequestMapping(valu ...

  10. jqgird

    将jqgird某字段设为超链接,并传递相关参数 cellvalue:为后台传递过来的字段数据 rowObject:为本行数据 实现: formatter:function(cellvalue, opt ...