LeetCode-4MedianofTwoSortedArrays(C#)
# 题目
4. 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 = [, ] nums2 = [] The median is 2.0
Example 2:
nums1 = [, ] nums2 = [, ] The median + )/ = 2.5
# 思路
两指针填充数组法:
想法来源于插入排序的过程。插入排序每一次将一位数字之前插入有序数组之中。
例如{1, 3, 4, 2},前面三位已用插入排序排序完毕,第四位2首先和1比较,大于,再和后面一位比较,小于,故放入该位置,其余元素后移。
此题的算法:
首先,把指针index1指向nums1,指针Index2指向nums2,同时创建一个数组sum,用来保存两个数组按照大小合并的结果。
接着,比较index1和index2所指元素的大小,sum数组填充两者较小的一个,同时指向较小元素的指针后移一位。
反复上面一步,当sum填充到中间时,结束,根据奇偶,计算中间值。
举个例子:
比如nums1 = {1, 3},nums2 = {2, 4, 5},index1指向nums1中的1,index2指向nums2中的2,1 < 2,所以sum填充1。
此时index1后移指向3,3 > 2,sum填充2,index2后移。此时index2指向4,4 > 3,sum填充3。此时sum已经填充到中部,故直接输出3。
// use two pointers to construct full nums including nums1 and nums2
// two pointers: time O(m+n) space O(n) result: 255ms
public double FindMedianSortedArrays(int[] nums1, int[] nums2)
{
int[] sum;
// nums1 is null or num2 is null
if (nums1.Length == 0) sum = nums2;
else if (nums2.Length == 0) sum = nums1;
else sum = new int[nums1.Length + nums2.Length];
// user 2 pointers to find middle numbers
int length = nums1.Length + nums2.Length, index1 = 0, index2 = 0, index = 0;
for (; index < sum.Length && index1 < nums1.Length && index2 < nums2.Length; index++)
{
if (nums1[index1] < nums2[index2]) sum[index] = nums1[index1++];
else sum[index] = nums2[index2++];
}
for (; index1 < nums1.Length; index++, index1++)
sum[index] = nums1[index1];
for (; index2 < nums2.Length; index++, index2++)
sum[index] = nums2[index2];
if (length % 2 != 0) return sum[length / 2];
else return (double)(sum[length / 2 - 1] + sum[length / 2]) / 2;
}
两指针填充数组法,时间复杂度O(m + n),空间复杂度O(n),时间255ms。
算法虽然AC(Accept)了,可并不满足题中所给的时间复杂度O(lg(m + n))。谈到lg,即以2为底的对数,很容易联想到对半的思想,二分查找法。
然而我并想不出来如何二分。
去LeetCode的Discuss区看了一下。发现了这样一个算法:二分查找,划分两数组法。参考地址:https://discuss.leetcode.com/topic/4996/share-my-o-log-min-m-n-solution-with-explanation/90。
天才。
只说说大概流程,具体内容直接参考链接,不懂请大胆留言。
算法的主要目的是将nums1和nums2形式意义上合并,再划分成两个规模相等且右侧所有数字大于左侧所有数字的数组(利用指针标记划分的位置),再根据划分的位置(由于两部分规模相等,必然是中间元素),输出结果。
如下图,我们先确定划分位置,这个划分位置一定会将左右两侧等分。

之后我们要保证左侧所有元素小于右侧所有元素,由于nums1和nums2是有序数组,边界元素就是划分之中最大/最小的元素,所以我们只需要关注边界即可。
假设nums1划分的位置是i,nums2划分的位置是j,要满足左侧元素小于右侧所有元素,必须满足nums1[i - 1] < nums1[i](数组有序,必定满足),nums1[i - 1] < nums2[j],nums1[i] > nums2[j - 1],nums2[j - 1] < nums2[j](数组有序,必定满足)。
若不满足nums1[i - 1] < nums2[j],如图,重新确定划分位置(图只是重划分的位置只是减少了1,实际上重划分的位置采用二分法)。

若不满足nums1[i] > nums2[j - 1],如图,重新确定划分位置(图只是重划分的位置只是减少了1,实际上重划分的位置采用二分法)。

反复上述过程,最后我们可以得到中间值。

# 解决(二分查找,划分两数组法)
// reference: https://discuss.leetcode.com/topic/4996/share-my-o-log-min-m-n-solution-with-explanation/
// time: O(lg(min(nums1,nums2)) space: O(1) result: 228ms
public double FindMedianSortedArrays(int[] nums1, int[] nums2)
{
&& nums2.Length == ) throw new Exception("none value in nums1 and nums2");
&& nums2.Length % == ) - ] + nums2[nums2.Length / ]) / ;
&& nums2.Length % != ) ];
&& nums1.Length % == ) - ] + nums1[nums1.Length / ]) / ;
&& nums1.Length % != ) ];
if (nums1.Length > nums2.Length) return FindMedianSortedArrays(nums2, nums1);
, imax = nums1.Length, midLength = (nums1.Length + nums2.Length + ) / ;
int i, j;
// when i isn't the split must satisfy below condition
do
{
i = (imin + imax) / ;
j = midLength - i;
&& j < nums2.Length && nums1[i - ] > nums2[j]) imax--; // nums[i - 1] > nums[j] means imin is too samll
&& nums1[i] < nums2[j - ]) imin++; // nums1[i] < nums2[j - 1] means imax is small
else break;
}
while (true);
// i is split, so we get answer by conditon
// get left first
int left;
) left = nums2[j - ];
) left = nums1[i - ];
] > nums2[j - ] ? nums1[i - ] : left = nums2[j - ];
// sum length is odd
!= )
{
return (double)left;
}
else
{
int right;
if (i == nums1.Length) right = nums2[j];
else if (j == nums2.Length) right = nums1[i];
else right = nums1[i] < nums2[j] ? nums1[i] : nums2[j];
;
}
}
二分查找,划分两数组法,时间复杂度O(lg(m + n)),空间复杂度O(1),时间228ms。
# 题外话
图够丑吗?来喷我啊,嘿嘿?
# 测试用例
static void Main(string[] args)
{
_4MedianofTwoSortedArrays solution = new _4MedianofTwoSortedArrays();
Debug.Assert(solution.FindMedianSortedArrays(, }, }) == 2.0d, "wrong 1"); // odd
Debug.Assert(solution.FindMedianSortedArrays( }, , }) == 2.0d, "wrong 2"); // odd
Debug.Assert(solution.FindMedianSortedArrays(, }, , }) == 2.0d, "wrong 3"); // even and duplication
Debug.Assert(solution.FindMedianSortedArrays(, }, , }) == 2.5d, "wrong 4"); // even
Debug.Assert(solution.FindMedianSortedArrays(, }) == 3.5d, "wrong 5"); // nums1 null
Debug.Assert(solution.FindMedianSortedArrays(, }, new int[] { }) == 2d, "wrong 6"); // nums2 numm
Debug.Assert(solution.FindMedianSortedArrays(, , , , , }, }) == 4d, "wrong 7"); // only nums1 pointer move
Debug.Assert(solution.FindMedianSortedArrays( }, , , , , , , }) == 4.5d, "wrong 8"); // only nums2 pointer move
Debug.Assert(solution.FindMedianSortedArrays(, , }, , , , , }) == 3.5d, "wrong 9");
Debug.Assert(solution.FindMedianSortedArrays(, }, , , , , }) == 3d, "wrong 10");
}
# 地址
Q: https://leetcode.com/problems/median-of-two-sorted-arrays/
A: https://github.com/mofadeyunduo/LeetCode/tree/master/4MedianofTwoSortedArrays
(希望各位多多支持本人刚刚建立的GitHub和博客,谢谢,有问题可以邮件609092186@qq.com或者留言,我尽快回复)
LeetCode-4MedianofTwoSortedArrays(C#)的更多相关文章
- 我为什么要写LeetCode的博客?
# 增强学习成果 有一个研究成果,在学习中传授他人知识和讨论是最高效的做法,而看书则是最低效的做法(具体研究成果没找到地址).我写LeetCode博客主要目的是增强学习成果.当然,我也想出名,然而不知 ...
- LeetCode All in One 题目讲解汇总(持续更新中...)
终于将LeetCode的免费题刷完了,真是漫长的第一遍啊,估计很多题都忘的差不多了,这次开个题目汇总贴,并附上每道题目的解题连接,方便之后查阅吧~ 477 Total Hamming Distance ...
- [LeetCode] Longest Substring with At Least K Repeating Characters 至少有K个重复字符的最长子字符串
Find the length of the longest substring T of a given string (consists of lowercase letters only) su ...
- Leetcode 笔记 113 - Path Sum II
题目链接:Path Sum II | LeetCode OJ Given a binary tree and a sum, find all root-to-leaf paths where each ...
- Leetcode 笔记 112 - Path Sum
题目链接:Path Sum | LeetCode OJ Given a binary tree and a sum, determine if the tree has a root-to-leaf ...
- Leetcode 笔记 110 - Balanced Binary Tree
题目链接:Balanced Binary Tree | LeetCode OJ Given a binary tree, determine if it is height-balanced. For ...
- Leetcode 笔记 100 - Same Tree
题目链接:Same Tree | LeetCode OJ Given two binary trees, write a function to check if they are equal or ...
- Leetcode 笔记 99 - Recover Binary Search Tree
题目链接:Recover Binary Search Tree | LeetCode OJ Two elements of a binary search tree (BST) are swapped ...
- Leetcode 笔记 98 - Validate Binary Search Tree
题目链接:Validate Binary Search Tree | LeetCode OJ Given a binary tree, determine if it is a valid binar ...
- Leetcode 笔记 101 - Symmetric Tree
题目链接:Symmetric Tree | LeetCode OJ Given a binary tree, check whether it is a mirror of itself (ie, s ...
随机推荐
- Angular2入门系列教程7-HTTP(一)-使用Angular2自带的http进行网络请求
上一篇:Angular2入门系列教程6-路由(二)-使用多层级路由并在在路由中传递复杂参数 感觉这篇不是很好写,因为涉及到网络请求,如果采用真实的网络请求,这个例子大家拿到手估计还要自己写一个web ...
- CoreCRM 开发实录 —— Profile
再简单的功能,也需要一坨代码的支持.Profile 的编辑功能主要就是修改个人的信息.比如用户名.头像.性别.电话--虽然只是一个编辑界面,但添加下来,涉及了6个文件的修改和7个新创建的文件.各种生成 ...
- KMP算法求解
// KMP.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h" #include<iostream> using namespac ...
- Java基础Map接口+Collections工具类
1.Map中我们主要讲两个接口 HashMap 与 LinkedHashMap (1)其中LinkedHashMap是有序的 怎么存怎么取出来 我们讲一下Map的增删改查功能: /* * Ma ...
- 构建通用的 React 和 Node 应用
这是一篇非常优秀的 React 教程,这篇文章对 React 组件.React Router 以及 Node 做了很好的梳理.我是 9 月份读的该文章,当时跟着教程做了一遍,收获很大.但是由于时间原因 ...
- Android 7.1 - App Shortcuts
Android 7.1 - App Shortcuts 版权声明:本文为博主原创文章,未经博主允许不得转载. 微博:厉圣杰 源码:AndroidDemo/Shortcuts 文中如有纰漏,欢迎大家留言 ...
- MJRefresh 源码解读 + 使用
MJRefresh这个刷新控件是一款非常好用的框架,我们在使用一个框架的同时,最好能了解下它的实现原理,不管是根据业务要求在原有的基础上修改代码,还是其他的目的,弄明白作者的思路和代码风格,会受益匪浅 ...
- Unity3D框架插件uFrame实践记录(一)
1.概览 uFrame是提供给Unity3D开发者使用的一个框架插件,它本身模仿了MVVM这种架构模式(事实上并不包含Model部分,且多出了Controller部分).因为用于Unity3D,所以它 ...
- 我这么玩Web Api(二):数据验证,全局数据验证与单元测试
目录 一.模型状态 - ModelState 二.数据注解 - Data Annotations 三.自定义数据注解 四.全局数据验证 五.单元测试 一.模型状态 - ModelState 我理解 ...
- CommandPattern
/** * 命令模式 * @author TMAC-J * 将调用者和接受者分离 * 可以将一组命令组合在一起,适合很多命令的时候 */ public class CommandPattern { i ...