http://blog.csdn.net/realxie/article/details/8078043

假设有长度分为为M和N的两个升序数组A和B,在A和B两个数组中查找第K大的数,即将A和B按升序合并后的第K个数。

解法一:

使用两个指针指向A和B的开头,很容易在O(M+N)的时间内完成,此算法略过。

解法二:

使用二分的方法。算法思想在代码注释中

  1. #include <iostream>
  2. #include <string.h>
  3. #include <stdlib.h>
  4. using namespace std;
  5. //Notice : K > 0
  6. int FindKthElm(int A[], int aBeg, int aEnd, int B[], int bBeg, int bEnd, int k)
  7. {
  8. if (aBeg > aEnd)
  9. {
  10. return B[bBeg + k - 1];
  11. }
  12. if (bBeg > bEnd)
  13. {
  14. return A[aBeg + k - 1];
  15. }
  16. //取中间位置
  17. int aMid = aBeg + (aEnd - aBeg)/2;
  18. int bMid = bBeg + (bEnd - bBeg)/2;
  19. //从A和B的开始位置到两个数组中间位置的元素个数
  20. int halfLen = aMid - aBeg + bMid - bBeg + 2;
  21. if (A[aMid] < B[bMid])
  22. {
  23. if (halfLen > k)
  24. {
  25. // 此时在合并的数组中A[aBeg...aMid]和元素一定在B[bMid]的左侧,
  26. // 即此时第k大的元素一定比B[bMid]这个元素小(严格来说不大于)
  27. // 故以后没有必要搜索 B[bMid...bEnd]这些元素
  28. return FindKthElm(A, aBeg, aEnd, B, bBeg, bMid - 1, k);
  29. }
  30. else
  31. {
  32. // 此时在合并的数组中A[aBeg...aMid]元素一定在B[bMid]的左侧,
  33. // 所以前K个元素中一定包含A[aBeg...aMid](可以使用反证法来证明这点)。
  34. // 但是无法判断A[amid+1...aEnd]与B[bBeg...bEnd]之间的关系,帮需要对他们进行判断
  35. // 此时K就剩下除去A[aBeg...aMid]这些元素,个数为k - (aMid - aBeg + 1)
  36. return FindKthElm(A, aMid + 1, aEnd, B, bBeg, bEnd, k - (aMid - aBeg + 1));
  37. }
  38. }
  39. else
  40. {
  41. //注释与上面相似
  42. if (halfLen > k)
  43. {
  44. return FindKthElm(A, aBeg, aMid - 1, B, bBeg, bEnd, k);
  45. }
  46. else
  47. {
  48. return FindKthElm(A, aBeg, aEnd, B, bMid + 1, bEnd, k - (bMid - bBeg + 1));
  49. }
  50. }
  51. }
  52. int main()
  53. {
  54. const int ALen = 11;
  55. const int BLen = 5;
  56. int apos = 0;
  57. int bpos = 0;
  58. int A[ALen];
  59. int B[ALen];
  60. //生成两个递增数组A 和 B
  61. for (int i = 1; i <= ALen + BLen; ++i)
  62. {
  63. if (apos >= ALen)
  64. {
  65. B[bpos++] = i;
  66. }
  67. else if (bpos >= BLen)
  68. {
  69. A[apos++] = i;
  70. }
  71. else
  72. {
  73. if (rand()%2 == 1)
  74. {
  75. A[apos++] = i;
  76. }
  77. else
  78. {
  79. B[bpos++] = i;
  80. }
  81. }
  82. }
  83. //输出A和B的内容
  84. for (int i = 0; i < ALen; ++i)
  85. {
  86. cout <<A[i] <<" ";
  87. }
  88. cout <<endl;
  89. for (int i = 0; i < BLen; ++i)
  90. {
  91. cout <<B[i] <<" ";
  92. }
  93. cout <<endl;
  94. //验证每个K是不是正解
  95. for (int i = 1; i <= ALen + BLen; ++i)
  96. {
  97. cout << i <<" : "<<FindKthElm(A, 0 , ALen - 1, B, 0 , BLen - 1, i)<<endl;
  98. }
  99. return 0;
  100. }

[转载]寻找两个有序数组中的第K个数或者中位数的更多相关文章

  1. 查找两个有序数组中的第K个元素(find kth smallest element in 2 sorted arrays)

    查找两个有序数组中的第K个元素 int FindKth(int a[], int b[], int k, int astart, int aend, int bstart, int bend) { ; ...

  2. 选取两个有序数组中最大的K个值,降序存入另一个数组中

    原题: 假设有两个有序的整型数组int *a1, int *a2,长度分别为m和n.试用C语言写出一个函数选取两个数组中最大的K个值(K可能大于m+n)写到int *a3中,保持a3降序,并返回a3实 ...

  3. 两个有序数组中查找第K大数

    题目:两个数组A.B,长度分别为m.n,即A(m).B(n),分别是递增数组.求第K大的数字.   方法一: 简单的办法,使用Merge Sort,首先将两个数组合并,然后在枚举查找.这个算法的时间复 ...

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

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

  5. 寻找两个有序数组的中位数 C++实现leetcode系列(四)

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

  6. Coursera Algorithms week3 快速排序 练习测验: Selection in two sorted arrays(从两个有序数组中寻找第K大元素)

    题目原文 Selection in two sorted arrays. Given two sorted arrays a[] and b[], of sizes n1 and n2, respec ...

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

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

  8. 两个有序数组中的中位数以及求第k个最小数的值

    解法参考 <[分步详解]两个有序数组中的中位数和Top K问题> https://blog.csdn.net/hk2291976/article/details/51107778 里面求中 ...

  9. 0004. 寻找两个有序数组的中位数(Java)

    4. 寻找两个有序数组的中位数 https://leetcode-cn.com/problems/median-of-two-sorted-arrays/ 最简单的就是用最简单的,把两个数组分别抽出然 ...

随机推荐

  1. 【转】CentOS中vsftp安装、配置、卸载

    1. 安装VSFTP yum -y install vsftpd 2. 配置vsftpd.conf文件 # Example config file /etc/vsftpd/vsftpd.conf # ...

  2. 体验CoreCLR的stack unwinding特性在Linux/Mac上的初步实现

    有了stack unwinding特性,才能在.NET程序中获取调用堆栈(call stack)信息,才能在异常时显示调用堆栈信息.这个特性之前只在Windows上有实现,Linux/Mac上的实现最 ...

  3. 博客搬家了,欢迎访问 http://blog.csdn.net/yinpengxiang/

    博客搬家了,欢迎访问 http://blog.csdn.net/yinpengxiang/

  4. SpringMVC实现一个controller写多个方法

    MultiActionController与ParameterMethodNameResolver在一个Controller类中定义多个方法,并根据使用者的请求来执行当中的某个方法,相当于Struts ...

  5. atitit.RESTful服务的概览and框架选型

    atitit.RESTful服务的概览and框架选型 1. REST基础概念: 1 2. URL说明: 1 3.  1 4. RESTful框架选型 2 1. spring mvc( recomm) ...

  6. paip.调试js 查看元素事件以及事件断点

    paip.调试js  查看元素事件以及事件断点 ff 26 +firebug 查看不出来.. 360 ,虽然也是chrome 基础,但是开发工具烂阿,也是显示不出来.. 作者Attilax  艾龙,  ...

  7. Linux初学 - 安装及网络配置

    安装版本 CentOS-6.4 虚拟机  vmware workstation 12 配置 网络配置 检查网络设置是否成功 如果网络配置文件检查没有问题,配置完成后网络仍然ping不同 1.检查虚拟机 ...

  8. maven源码分析- mvn.bat分析

    第一次知道MAVEN是在2008年,当时想分析geoserver这个开源项目,发现该项目采用了maven进行项目管理,当时粗略的学习了一下.真正在工作中使用是在09年下半年,个人感觉使用起来还是非常好 ...

  9. Android PullToRefresh (ListView GridView 下拉刷新) 使用详解 (转载)

    最近项目用到下拉刷新,上来加载更多,这里对PullToRefresh这控件进行了解和使用. 以下内容转载自:http://blog.csdn.net/lmj623565791/article/deta ...

  10. Android布局优化

    前言 本篇文章为Android优化的布局部分,该部分应该是Android中很重要的,无论是在自定义控件中,还是在简单的书写布局时,都应该尽量遵循一些优化原则,这样布局的绘制效率才会更高,体验才能更好. ...