LeetCode第[4]题(Java):Median of Two Sorted Arrays (俩已排序数组求中位数)——HARD
题目难度:hard
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
翻译:
有两个已排序的数组nums1和nums2长度分别是m和n。
找到这两个数组的中值。整个运行时复杂度应该是O(log(m+n))。
想了老半天,
思路一:将两个数组直接放入List,然后调用Collection.sort()排好,最后返回其中位值
思路二:利用归并排序中的合并算法,将两个数组按顺序比大小拼接成一个数组(),最后返回其中位值
思路一Code:2080测试用例—107ms(beats 6.25%) 时间复杂度:O(N*logN)…………假设Collections.sort()为快速排序O(N*logN)
public double findMedianSortedArrays(int[] nums1, int[] nums2) {
List<Integer> list = new ArrayList<Integer>();
for (int i = 0; i < nums1.length; i++) {
list.add(nums1[i]);
}
for (int i = 0; i < nums2.length; i++) {
list.add(nums2[i]);
}
Collections.sort(list);
if ( list.size() % 2 == 0) {
return (list.get(list.size()/2) + list.get(list.size()/2 - 1))/2.0;
} else {
return list.get(list.size()/2);
}
}
用了Collection自带的sort方法,省得自己写排序代码。。。呃,有种作弊的感脚
总体速度貌似也还行,就是显得蠢了点=。=根本没用到“已排序”这个条件
【最近发现java.util.Arrays类直接就有Arrays.sort(int[] a)方法。。。】
思路二Code:2080测试用例—66ms(beats 70.29%) 时间复杂度:O(N+M)…………最小为O(N)
public static double findMedianSortedArrays2(int[] A, int[] B) {
int[] uniArray = uniSort(A, B);
if (uniArray.length % 2 == 0) {
return (uniArray[uniArray.length / 2] + uniArray[uniArray.length / 2 - 1]) / 2.0;
} else {
return uniArray[uniArray.length / 2];
}
} private static int[] uniSort(int[] A, int[] B) {
int[] result = new int[A.length + B.length];
int i = 0,j = 0;
int k = 0; while (i < A.length && j < B.length) {
if (A[i] < B[j]) {
result[k++] = A[i++];
} else {
result[k++] = B[j++];
}
}
while (i < A.length) {
result[k++] = A[i++];
}
while (j < B.length) {
result[k++] = B[j++];
}
return result;
}
这种方法应该是最容易想到的了,没啥技巧性可言,就是用一个辅助数组和一个int k 作为指针进行移动
代码可读性不错,时间复杂度也可观。
好吧,还是显得蠢了点。。。
参考答案(选了个可读性好点的)
思想:采用二分法,不考虑数值大小,只管查找当前下标是否在需要定位的下标的一半内
Code:2080测试用例—86ms(beats 21.34%) 时间复杂度:O(log(m + n))
public static double findMedianSortedArrays(int[] nums1, int[] nums2) {
// 处理无效边界
if (nums1 == null || nums2 == null) return 0.0; int m = nums1.length, n = nums2.length;
if ((m + n) % 2 !=0) {
// 如果 m + n 长度是奇数,返回中间那一个
return getKth(nums1, 0, nums2, 0, (m + n + 1) / 2);
} else {
// 如果 m + n 长度是偶数,两个函数将返回一个左数和一个右数
return (getKth(nums1, 0, nums2, 0, (m + n + 1) / 2) + getKth(nums1, 0, nums2, 0, (m + n + 2) / 2)) / 2.0;
}
} private static double getKth(int[] nums1, int start1, int[] nums2, int start2, int k) {
// 这个函数旨在nums1+nums2中找到第k个元素[而不是下标为k,这两个数组并没有合并不存在统一下标] // 如果nums1耗尽,则返回nums2中的kth号
if (start1 > nums1.length - 1) return nums2[start2 + k - 1]; // 如果nums2耗尽,则返回nums1中的第k号
if (start2 > nums2.length - 1) return nums1[start1 + k - 1]; // 如果k==1,返回第一个数字
// 因为nums1和nums2是排序的,所以nums1和nums2的起始点中的较小的一个是第一个
if (k == 1) return Math.min(nums1[start1], nums2[start2]); int mid1 = Integer.MAX_VALUE;
int mid2 = Integer.MAX_VALUE;
// 为什么不取0,因为当某一边长度不够折半时,它的mid默认长度应该比任何能折半的mid都大(保证让对方折半)
if (start1 + k / 2 - 1 < nums1.length) mid1 = nums1[start1 + k / 2 - 1];
if (start2 + k / 2 - 1 < nums2.length) mid2 = nums2[start2 + k / 2 - 1]; // 将数组中的一半从nums1或nums2中删除。把k切成两半
if (mid1 < mid2) {
return getKth(nums1, start1 + k / 2, nums2, start2, k - k / 2); //nums1.right + nums2
} else {
return getKth(nums1, start1, nums2, start2 + k / 2, k - k / 2); //nums1 + nums2.right
}
}
看了老半天才勉强看懂,比较难理解,不过时间复杂度小。
下面举个例子方便理解:
A【。。。】数组
B【。。。】数组
很假设中位数下标为k
下面以这俩数组为例,介绍这个“二分法求俩已排序数组中位数”
1. k =( n + m +1) / 2 // 假设m+n是奇数
2. 假设合并后的数组为C【。。。】,现在将C平均分成4部分,所以可以分为
C【0,k/2-1】;// k/2-1前半段的中位数的下标
C【略1】
C【略2】
C【略3】
3. 现在假如A内可以数到下标 k/2-1 = 1,B也可以数到。
所以C的k点的值必然大于 A数组k/2 与 B数组k/2 这两点中小的那一个点!所以小的那一个点对应的数组从起点到此点都不可能出现k,所以可以删去。
所以比较 A数组k/2 与 B数组k/2 ,将小的那个数组从start到k/2截去(将起点设置为start+k/2-1),然后继续递归搜索。
总结:如果真的出了这种题目,笔试就写第二种吧,复杂度也差不太多。面试的话先说第二种,再说第三种。
ps:LeetCode中写注释貌似会影响最后的成绩。
LeetCode第[4]题(Java):Median of Two Sorted Arrays (俩已排序数组求中位数)——HARD的更多相关文章
- 2.Median of Two Sorted Arrays (两个排序数组的中位数)
要求:Median of Two Sorted Arrays (求两个排序数组的中位数) 分析:1. 两个数组含有的数字总数为偶数或奇数两种情况.2. 有数组可能为空. 解决方法: 1.排序法 时间复 ...
- Leetcode4.Median of Two Sorted Arrays两个排序数组的中位数
给定两个大小为 m 和 n 的有序数组 nums1 和 nums2 . 请找出这两个有序数组的中位数.要求算法的时间复杂度为 O(log (m+n)) . 你可以假设 nums1 和 nums2 不同 ...
- leetcode第四题:Median of Two Sorted Arrays (java)
Median of Two Sorted Arrays There are two sorted arrays A and B of size m and n respectively. Find t ...
- [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 ...
- 【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 tw ...
- [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 ...
- [LeetCode] 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 ...
- 004 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 ...
- 4. Median of Two Sorted Arrays(2个有序数组的中位数)
There are two sorted arrays nums1 and nums2 of size m and n respectively. Find the median of the two ...
随机推荐
- Oracle数据库的连接模式connection Mode、连接connection与会话session
数据库的连接模式Connection Mode: Dedicated Server Mode(专有模式) 当用户发出请求时,如远程的client端通过监听器连接数据库上,ORACLE的服务器端会启用一 ...
- mysql 获取id最大值
数据库表中id列不为自动增加,需要程序来增加id的SQL SELECTCASE IFNULL(MAX(id),1)WHEN 1 THEN 1ELSE MAX(id) + 1END AS newmaxi ...
- 个案排秩 Rank (linear algebra) 秩 (线性代数)
非叫“秩”不可,有秩才有解_王治祥_新浪博客http://blog.sina.com.cn/s/blog_8e7bc4f801012c23.html 我在一个大学当督导的时候,一次我听一位老师给学生讲 ...
- IO流入门-第八章-BufferedWriter
BufferedWriter基本用法和方法示例 import java.io.*; public class BufferedWriterTest01 { public static void mai ...
- appium入门基础知识
1.概念区分: 1)IOS-UIAutomation:随着iOS4.0的发布,苹果公司同时发布了一个名为UIAutomation的测试框架,它可以用来在真实设备和iPhone模拟器上执行自动化测试 学 ...
- 我的Android进阶之旅------>android Button上面的英文字符串自动大写的问题解决
今天碰到一个关于Button的问题:android Button上面的英文字符串会自动变成大写,运行的Android 5.1版本,如下图所示: 图1:Button 图2:TextView 这个Butt ...
- 斯坦福大学Andrew Ng - 机器学习笔记(8) -- 推荐系统 & 大规模机器学习 & 图片文字识别
大概用了一个月,Andrew Ng老师的机器学习视频断断续续看完了,以下是个人学习笔记,入门级别,权当总结.笔记难免有遗漏和误解,欢迎讨论. 鸣谢:中国海洋大学黄海广博士提供课程视频和个人笔记,在此深 ...
- 000 初步使用Kotlin开发Android应用
Kotlin是Jetbrians公司开发的一款编程语言,基于jvm兼容Java. 要求 IDE:IDEA或者Android Studio(简称studio)对Kotlin语言有所了解,官方文档:htt ...
- Angular学习笔记—路由(转载)
创建路由 1.首先安装 Angular Router.你可以通过运行以下任一操作来执行此操作: yarn add @angular/router # OR npm i --save @angular/ ...
- 对象序列化与反序列化local class incompatible
无论eclipse还是idea(默认关闭序列化提示,需手动打开),都可以自动生成相应的序列号,分为两类1L,XXXL. 当然如果不指定,系统也会自动生成,但是存在隐性风险 ,不同的编译器对同一个对象可 ...