leetcode-4. 寻找两个正序数组的中位数。

给定两个大小为 m 和 n 的正序(从小到大)数组 nums1 和 nums2。

请你找出这两个正序数组的中位数,并且要求算法的时间复杂度为 O(log(m + n))。

你可以假设 nums1 和 nums2 不会同时为空。

示例 1:

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

则中位数是 2.0

示例 2:

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

则中位数是 (2 + 3)/2 = 2.5

来源:力扣(LeetCode)

链接:https://leetcode-cn.com/problems/two-sum

著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

解法一: 辅助数组对原来的两数组进行归并。
double findMedianSortedArrays( int *nums1, int nums1Size, int *nums2, int nums2Size ) {
	double answer = 0.0;
	int i1 = 0, i2 = 0, ti = 0, *tempArray = NULL;

	tempArray = malloc( sizeof(*tempArray) * (nums1Size + nums2Size) );

	while( i1 < nums1Size && i2 < nums2Size ) {
		tempArray[ti++] = nums1[i1] <= nums2[i2] ? nums1[i1++] : nums2[i2++];
	}
	while( i1 < nums1Size ) {
		tempArray[ti++] = nums1[i1++];
	}
	while( i2 < nums2Size ) {
		tempArray[ti++] = nums2[i2++];
	}

	answer = (ti & 1) ? tempArray[ti / 2] * 2 : tempArray[ti / 2] + tempArray[ti / 2 - 1];

	free( tempArray );

	return answer / 2.0;
}
解法二:模拟归并过程。

其实并不需要将两数组合并, 只需要找到中位数在位置即可.

double findMedianSortedArrays( int *nums1, int nums1Size, int *nums2, int nums2Size ) {
	int lasttime = 0, current = 0, len = nums1Size + nums2Size;
	int half = len / 2;

	for( int i1 = 0, i2 = 0, i = 0; i <= half; ++i ) {
		lasttime = current; // 记录中位数上一次的位置的值.
		if( i2 >= nums2Size || (i1 < nums1Size && nums1[i1] <= nums2[i2]) ) {
			current = nums1[i1++];
		} else if( i2 < nums2Size ) {
			current = nums2[i2++];
		}
	}

	return len & 1 ? current : (lasttime + current) / 2.0;
}
解法三:求中位数其实就是求第 k 小数的一种特殊情况。

前面解法的遍历过程中每次只去掉一个不可能是中位数的元素,
由于数列是有序的,其实完全可以一半一半的排除,
假设要找第 k 小的元素, 遍历过程中每次排除掉 k/2 个元素.
参考:
https://leetcode-cn.com/problems/median-of-two-sorted-arrays/solution/xiang-xi-tong-su-de-si-lu-fen-xi-duo-jie-fa-by-w-2/

#define MIN( a, b ) ((a) <= (b) ? (a) : (b))

static int recursion( int a[], int aSize, int b[], int bSize, int k ) {
	int kc2 = k / 2;
	int ai = MIN( aSize, kc2 ) - 1, bi = MIN( bSize, kc2 ) - 1;

	if( aSize < 1 || bSize < 1 ) {
		return aSize < 1 ? b[k - 1] : a[k - 1];
	}
	if( k < 2 ) {
		return MIN( a[0], b[0] );
	}
	if( a[ai] <= b[bi] ) {
		return recursion( a + MIN( aSize, ai + 1 ), aSize - (ai + 1), b, bSize, k - (ai + 1) );
	}

	return recursion( a, aSize, b + MIN( bSize, bi + 1 ), bSize - (bi + 1), k - (bi + 1) );
}

double findMedianSortedArrays( int *nums1, int nums1Size, int *nums2, int nums2Size ) {
	int len = nums1Size + nums2Size;
	double answer = 0.0;

	// 假设数组共有7个元素, 则:
	// 中位数 = (第4个元素 + 第4个元素) / 2.0.
	// 中位数 = (第(8/2)个元素 + 第(9/2)个元素) / 2.0.

	// 假设数组共有8个元素, 则:
	// 中位数 = (第4个元素 + 第5个元素) / 2.0.
	// 中位数 = (第(9/2)个元素 + 第(10/2)个元素) / 2.0.

	// 数组的元素数量不管是奇数还是偶数,
	// 中位数都为 (第((len+1)/2)个元素 + 第((len+2)/2)个元素) / 2.0.

	answer += recursion( nums1, nums1Size, nums2, nums2Size, (len + 1) / 2 );
	answer += recursion( nums1, nums1Size, nums2, nums2Size, (len + 2) / 2 );

	return answer / 2.0;
}

leetcode-4. 寻找两个正序数组的中位数的更多相关文章

  1. 寻找两个已序数组中的第k大元素

    寻找两个已序数组中的第k大元素 1.问题描述 给定两个数组与,其大小分别为.,假定它们都是已按照增序排序的数组,我们用尽可能快的方法去求两个数组合并后第大的元素,其中,.例如,对于数组,.我们记第大的 ...

  2. 【算法剖析】寻找两个已序数组中的第k大元素

    1.问题描述 给定两个数组A与B,其大小分别为m.n,假定它们都是已按照增序排序的数组,我们用尽可能快的方法去求两个数组合并后第k大的元素,其中,1\le k\le(m+n).例如,对于数组A=[1, ...

  3. Java实现 LeetCode 4 寻找两个有序数组的中位数

    寻找两个有序数组的中位数 给定两个大小为 m 和 n 的有序数组 nums1 和 nums2. 请你找出这两个有序数组的中位数,并且要求算法的时间复杂度为 O(log(m + n)). 你可以假设 n ...

  4. 【LeetCode】寻找两个有序数组的中位数【性质分析+二分】

    给定两个大小为 m 和 n 的有序数组 nums1 和 nums2. 请你找出这两个有序数组的中位数,并且要求算法的时间复杂度为 O(log(m + n)). 你可以假设 nums1 和 nums2 ...

  5. [LeetCode] 4. 寻找两个有序数组的中位数

    题目链接:https://leetcode-cn.com/problems/median-of-two-sorted-arrays/ 题目描述: 给定两个大小为 m 和 n 的有序数组 nums1 和 ...

  6. 【LeetCode】寻找两个有序数组的中位数

    给定两个大小为 m 和 n 的有序数组 nums1 和 nums2. 请你找出这两个有序数组的中位数,并且要求算法的时间复杂度为 O(log(m + n)). 你可以假设 nums1 和 nums2  ...

  7. leetcode 4寻找两个有序数组的中位数

    最优解O(log(min(m,n))) /** 之前用合并有序数组的思想做了O((m+n+1)/2),现在试一试O(log(min(m,n))) 基本思路为:通过二分查找较小的数组得到对应的中位数(假 ...

  8. LeetCode Golang 4. 寻找两个有序数组的中位数

    4. 寻找两个有序数组的中位数 很明显我偷了懒, 没有给出正确的算法,因为官方的解法需要时间仔细看一下... func findMedianSortedArrays(nums1 []int, nums ...

  9. Leetcode(4)寻找两个有序数组的中位数

    Leetcode(4)寻找两个有序数组的中位数 [题目表述]: 给定两个大小为 m 和 n 的有序数组 nums1 和* nums2. 请你找出这两个有序数组的中位数,并且要求算法的时间复杂度为 O( ...

随机推荐

  1. Shell命令和流程控制

    Shell命令和流程控制 在shell脚本中可以使用三类命令: 1)Unix 命令: 虽然在shell脚本中可以使用任意的unix命令,但是还是由一些相对更常用的命令.这些命令通常是用来进行文件和文字 ...

  2. RCC BUCK-BOOST变压器设计

    电路处于Buck-Boost DCM模式,最恶劣情况为输入电压最低时.取频率为f 以下图为例: 1.MOS耐压选择: Vmos≥Vinmax+Vout+80 2.保证磁芯不饱和且始终工作在DCM模式 ...

  3. css技术和实例

    今天,我为大家收集精选了30个使用纯CSS完成的强大实践的优秀CSS技术和实例,您将在这里发现很多与众不同的技术,比如:图片集.阴影效果.可扩展按钮.菜单等-这些实例都是使用纯CSS和HTML实现的. ...

  4. Android TextView标签的显示

    在默认情况下,如果一个TextView中的文字太多,会跨行显示, 通过下面两个参数的设置,可以使TextView固定显示一行,未显示完成的后面用...... android:maxLines=&quo ...

  5. leetcode Largest Rectangle in Histogram 解法二

    上一篇文章讲了该题的一个解法.后来又发现一个更好的解法. 首先依旧考虑一个升序的数列,例如1,2,3,4,5.那么它的最大矩形显然是有5种可能,即 1*5,2*4,3*3,4*2,1*5.所以最大的矩 ...

  6. codevs 3693 数三角形

    /* n*m个点中选3个 再排除三点共线 共线分两类 1 在横线或者竖线上 m*C(n,3) n*C(m,3) 2 在对角线上 这个比较麻烦 以为对角线和矩阵是一一对应的 我们转化成求矩阵 并且保证有 ...

  7. CCI_chapter 3 Stacks and Queues

    3.1Describe how you could use a single array to implement three stacks for stack 1, we will use [0, ...

  8. 201521123013 《Java程序设计》第8周学习总结

    1. 本章学习总结 2. 书面作业 Q1.List中指定元素的删除(题目4-1) 1.1 实验总结 while(list.contains(str)) list.remove(str); Q2.统计文 ...

  9. app.config 配置多项 配置集合 自定义配置(2)

    上一篇说了利用app.config自定义节点配置,那是利用工具来实现,其实也一全部编码的方式来实现.举一个栗子.Simpson一家有父亲James,母亲Kate,和三个儿女Jim,Aaron和Luka ...

  10. 用memset设置无穷大无穷小

    memeset是以字节为单位进行赋值的,对字符数组可以直接用. 但对于int数组就不行了. 但设置无穷大来说有个技巧: 如果我们将无穷大设为0x3f3f3f3f,那么奇迹就发生了,0x3f3f3f3f ...