`
eriol
  • 浏览: 410002 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

在两个有序数组中找第N数

阅读更多

给定两个有序的数组,长度分别为m和n,求这两个数组中的第K个元素。

 

 

问题分析:

 

1. 把 A 平均分为前后两个部分,前部分有 x 个元素,后部分有 n-x 个元素(由于 A 是有序的,所以后一部分的所有元素都大于前一部分)。A[x] 为 A 的后一部分中的第一个元素。

 

2. 同理把 B 也平均分成前后两个部分,前部分有 y 个元素,后部分有 m-y 个元素。B[y] 是 B 的后一部分中的第一个元素。

 

3. 由于两个数组都是被平均分割的,所以可以近似地认为 x = n/2, y = m/2。这里不妨设 A[x] <= B[y](如果 A[x] > B[y] 处理过程和下面类似): 

 

 

情况1:

 

由于在 A 中,A[x] 前面有 x 个元素,在 B 中,B[y] 前面有 y 个元素,并且又有 A[x] <= B[y],那么,合并以后,A[x]前面原来那些元素必然也在B[y]前面。也就是说,B[y]前面至少会有 x + y 个元素,我们再规定如果 A, B 中有相同元素,则合并后 A 中的元素排在 B 前面,那么归并以后 A[x] 也会排在 B[y] 前面,于是,合并之后 B[y] 至少有 x+y+1 个元素。 

 

如果 k <= x+y+1,也就是说,合并后第 k 个元素必然落在 B[y] 前面。所以,原来在 B 数组中的后一部分(B[y]以及 B[y] 之后)那些元素都不可能包含我们要找到内容(第 k 大元素),所以我们可以把他们排除掉。这样就排除了 B 中一半的内容。 

 


情况2:

 

在 A 中,A[x] 及其后面有 n-x 个元素,除去 A[x] 之后有 n-x-1 个元素,B[y] 及其后面有 m-y 个元素。那么,由于 A[x] <= B[y],所以合并起来之后,B[y] 后面那些元素必然也在 A[x] 后面,则合并后 A[x] 后面至少有(n-x-1) + (m-y) = (n+m)-(x+y+1) 个元素。 

 

如果 k > x+y+1,也就说,合并后第 k 大的元素必然落在 A[x] 后面。所以,原来在 A 数组中,第一部分(A[x]之前)以及 A[x] 都不可能包含我们要找的元素,所以我们可以把他们排除掉。这样就排除了 A 中一半的内容。 

 

 

总结:

 

综上所述,对于 k <= x+y+1 还是 k > x+y+1 我们都提出了解决的方案,并且每种方案都能把 A 或者 B 的规模减小一半。减小了一半之后,我们将其作为一个新的问题继续使用上面的算法处理,直到 A 或者 B 减小到足够小: 

  1. A 没有了,这样只需要找出 B 中第 k 大的元素,也就是 B[k]。
  2. B 没有了,同上结果就是 A[k]。

public class NumX {

	private int[] a;
	private int[] b;
	
	public int find(int aLeft, int aRight, int bLeft, int bRight, int k) {
		int aMid = (aLeft + aRight) / 2;
		int bMid = (bLeft + bRight) / 2;
		
		if (aLeft > aRight)
			return b[bLeft+k-1];
		if (bLeft > bRight)
			return a[aLeft+k-1];
		
		if (a[aMid] <= b[bMid]) {
			if (k <= (aMid-aLeft) + (bMid-bLeft) + 1)
				return find(aLeft, aRight, bLeft, bMid-1, k);
			else
				return find(aMid+1, aRight, bLeft, bRight, k-(aMid-aLeft)-1);
		} else {
			if (k <= (aMid-aLeft) + (bMid-bLeft) + 1)
				return find(aLeft, aMid-1, bLeft, bRight, k);
			else
				return find(aLeft, aRight, bMid+1, bRight, k-(bMid-bLeft)-1);
		}
	}
	
	public static void main(String[] args) {
		NumX n = new NumX();
		n.a = new int[] {1, 4, 6};
		n.b = new int[] {5, 8, 9};
		System.out.println(n.find(0, 2, 0, 2, 3));
	}
}

 

 

原文地址:http://blog.csdn.net/linandixon/article/details/4647392

分享到:
评论

相关推荐

    C++算法:寻找两个有序数组的中位数

    请你找出这两个有序数组的中位数,并且要求算法的时间复杂度为 O(log(m + n))。 你可以假设 nums1 和 nums2 不会同时为空。 示例 1: nums1 = [1, 3] nums2 = [2] 则中位数是 2.0 示例 2: nums1 = [1, 2] ...

    将两个有序数组,合并成另一个有序的数组,升序

    在计算机科学和编程领域中,将两个有序数组合并成另一个有序数组是一个经典的算法问题。这个问题不仅在理论学习中占有重要地位,而且在实际应用中也非常普遍。对于这个任务,核心目标是将两个已经按照升序排列的整数...

    分治法求两列有序数组的中位数的程序

    (1)设X[0:n-1]和Y[0:n-1]为两个数组,每个数组中含有n个已排好序的数,设计一个算法复杂度为O(logn)的分治算法,找出X和Y中2n个数中的中位数。(中位数:个数为奇数:中间位置上的数;个数为偶数,中间两个数的...

    高效合并两个有序数组.py

    输入参数nums1和nums2分别为两个数组,m和n表示nums1中待处理部分的长度和nums2的剩余长度。函数的核心是while循环,条件是两个数组还有元素未合并。 循环内部的判断是根据两个数组当前末尾元素的大小来决定将哪个...

    17082 两个有序数序列中找第k小

    17082 两个有序数序列中找第k小(必做) 时间限制:1000MS 内存限制:65535K 提交次数:0 通过次数:0 题型: 编程题 语言: C++;C;VC;JAVA Description 已知两个已经排好序(非减序)的序列X和Y,其中X的长度为m,Y长度为...

    两个有序数组求交集,C++

    ### 两个有序数组求交集(C++) 在计算机科学领域,处理数组的交集问题是一项基本而重要的任务。本文将详细介绍如何使用C++语言来实现两个有序数组的交集操作,并深入探讨其背后的原理和算法优化策略。 #### 1. 问题...

    ZYZMZM#LeetCode#4. 寻找两个有序数组的中位数1

    4. 寻找两个有序数组的中位数给定两个大小为 m 和 n 的有序数组 nums1 和 nums2。请你找出这两个有序数组的中位数,并且要求算法的时间复杂度为 O

    c语言编程题之数组操作合并两个有序数组.zip

    在上述代码中,`merge`函数实现了合并两个有序数组的功能。`printArray`函数用于打印数组,方便我们验证结果。`main`函数中定义了两个有序数组,并调用`merge`函数进行合并,最后输出合并后的数组。 通过这个编程题...

    前端JS合并两个有序数组,快速搞定!

    在合并两个有序数组时,我们希望保持合并后的数组依然有序。一种直观的方法是遍历两个数组,比较每个元素,然后将较小的元素添加到新数组中。这种方法虽然简单,但效率较低,时间复杂度为 O(n*m),其中 n 和 m 分别...

    C++实现两个有序数组的合并

    在本篇文章中,我们将详细介绍如何使用C++语言实现两个有序数组的合并。数组合并是数据结构和算法中的一种常见操作,掌握数组合并的技巧对于提高编程技能非常重要。 数组合并的概念 数组合并是指将两个或多个数组...

    java-leetcode题解之第88题合并两个有序数组.zip

    标题中的“java-leetcode题解之第88题合并两个有序数组”指的是LeetCode平台上的一个编程挑战,问题编号为88,它涉及到如何合并两个已排序的整数数组,使得结果仍然保持有序。LeetCode是一个热门的在线编程练习平台...

    算法/编程练习:两个有序数组的中位数

    找出这两个有序数组的中位数mid。 要求算法的时间复杂度为 O(log(m + n))。 例如, 输入: nums1 = [1, 3, 5, 7, 9], nums2 = [2, 4, 6, 8, 10, 11] 输出: mid=6 思路: 记总的数组长度为 N = n1 + n2,则两个数组...

    php-leetcode题解之合并两个有序数组.zip

    在本压缩包“php-leetcode题解之合并两个有序数组.zip”中,主要涉及的是一个经典的算法问题——如何使用PHP解决LeetCode上的“合并两个有序数组”问题。这是一个常见的计算机科学与编程面试题目,旨在考察程序员...

    合并两个有序数组1

    合并两个有序数组是计算机科学中一个常见的操作,特别是在数据结构和算法的学习中。这个任务主要涉及到数组的操作和排序。在LeetCode中,这个问题被称为“合并两个有序数组1”(Merge Two Sorted Arrays)。以下是对...

    python 两个排序数组的中位数

    # 请找出这两个有序数组的中位数。要求算法的时间复杂度为 O(log (m+n)) # 你可以假设 nums1 和 nums2 不同时为空 # 示例 1: # nums1 = [1, 3] # nums2 = [2] # 中位数是 2.0 # 示例 2: # nums1 = [1, 2] # nums2 ...

    寻找两个正序数组的中位数

    我们需要在两个有序数组中找到中位数,确保算法的时间复杂度为 O(log(min(m, n)))。 解题思路 确保第一个数组较小:始终在较小的数组上进行二分查找,以确保时间复杂度最优。 进行二分查找: 定义分割点 i 在 nums1...

    两个有序数序列中找第k小(必做)

    已知两个已经排好序(非减序)的序列X和Y,其中X的长度为m,Y长度为n, 现在请你用分治算法,找出X和Y的第k小的数,算法时间复杂度为O(max{logm, logn})。

    合并两个有序数组(java代码).docx

    ### 合并两个有序数组(Java代码) #### 知识点概述 本篇文章主要介绍了一种使用Java语言来实现合并两个有序整数数组的方法。在实际应用中,尤其是在数据处理和算法设计领域,有序数组的合并是一个非常实用且常见...

    java-leetcode面试题解双指针之第88题合并两个有序数组.zip

    总之,本压缩包中的资源提供了针对LeetCode第88题的Java解法,该题主要涉及双指针技术在合并两个有序数组问题上的应用。学习和理解这种解法,不仅能提升你在算法方面的技能,也有助于你在实际工作中的问题解决能力。

    数组中数对差最大

    数对由数组中的两个不同元素组成,其中一个元素位于另一个元素的右侧。计算差值是将左侧元素减去右侧元素,而不是相反,因为通常较小的数会被较大的数减去以得到较大的差值。 解决此类问题的一个常见方法是使用双...

Global site tag (gtag.js) - Google Analytics