问题描述

给定一个包含n个数的乱序数组,寻找其中第k大的数。

方法总结

解法1:先将乱序数组按照从大到小排序,然后取出第k大的数。时间复杂度为O(n*logn)。
解法2:利用选择排序,k次选择后即可得到第k大的数。时间复杂度为O(n*k)。
解法3:利用快速排序的思想,从数组S中随机找出一个元素X,把数组分为两部分Sa和Sb。Sa中的元素大于等于X,Sb中元素小于X。这时有两种情况:
1. Sa中元素的个数小于k,则Sb中的第k-|Sa|个元素即为第k大的数;
2. Sa中元素的个数大于等于k,则返回Sa中的第k大数。
时间复杂度近似为O(n)。
解法4:二分[Smin,Smax]查找结果X,统计X在数组中的出现次数m,以及数组中比X大的数目y,满足y<k且y+m>=k的数即为第k大数。时间复杂度为O(n*logn)。
解法5:用O(4*n)的方法对原数组建最大堆,然后pop出k次即可。时间复杂度为O(4*n + k*logn)。
解法6:维护一个大小为k的最小堆,对于数组中的每一个元素判断其与堆顶的大小,若堆顶较大,则不管;否则,弹出堆顶,将当前值插入到堆中。时间复杂度O(n*logk)。
解法7:利用hash表保存数组中元素Si出现的次数,利用基数排序的思想,线性从大到小扫描过程中,前面有k-1个数则为第k的大数,时间复杂度为O(n)。

相关问题及实际应用:

1. STL中可以用nth_element求得类似的第k大的数(由谓词决定),使用的是解法3中的思想,还可以用partial_sort对区间进行部分排序,得到类似前k大的数(由谓词决定),它采用的是解法5的思想。

2. 求中位数实际上是第k大数的特例。
《编程之美》2.5节课后习题:
(1). 如果需要找出N个数中最大的k个不同的浮点数呢?比如,含有10个浮点数的数组(1.5,1.5,2.5,3.5,3.5,5,0,- 1.5,3.5)中最大的3个不同的浮点数是(5,3.5,2.5)。
解答:上面的解法均适用,需要注意的是浮点数比较时和整数不同,另外求hashkey的方法也会略有不同。
(2). 如果是找第k到第m(0<k<=m<=n)大的数呢?
解答:如果把问题看做m-k+1个第k大问题,则前面解法均适用。但是对于类似前k大这样的问题,最好使用解法5或者解法7,总体复杂度较低。

3. 在搜索引擎中,网络上的每个网页都有“权威性”权重,如page rank。如果我们需要寻找权重最大的k个网页,而网页的权重会不断地更新,那么算法要如何变动以达到快速更新(incremental update)并及时返回权重最大的k个网页?

提示:堆排序?当每一个网页权重更新的时候,更新堆。还有更好的方法吗?
解答:要达到快速的更新,我们可以解法5,使用映射二分堆,可以使更新的操作达到O(logn)

4. 在实际应用中,还有一个“精确度”的问题。我们可能并不需要返回严格意义上的最大的k个元素,在边界位置允许出现一些误差。当用户输入一个query的时候,对于每一个文档d来说,它跟这个query之间都有一个相关性衡量权重f (query, d)。搜索引擎需要返回给用户的就是相关性权重最大的k个网页。如果每页10个网页,用户不会关心第1000页开外搜索结果的“精确度”,稍有误差是可以接受的。比如我们可以返回相关性第10 001大的网页,而不是第9999大的。在这种情况下,算法该如何改进才能更快更有效率呢?网页的数目可能大到一台机器无法容纳得下,这时怎么办呢?
提示:归并排序?如果每台机器都返回最相关的k个文档,那么所有机器上最相关k个文档的并集肯定包含全集中最相关的k个文档。由于边界情况并不需要非常精确,如果每台机器返回最好的k’个文档,那么k’应该如何取值,以达到我们返回最相关的90%*k个文档是完全精确的,或者最终返回的最相关的k个文档精确度超过90%(最相关的k个文档中90%以上在全集中相关性的确排在前k),或者最终返回的最相关的k个文档最差的相关性排序没有超出110%*k。
解答:正如提示中所说,可以让每台机器返回最相关的k'个文档,然后利用归并排序的思想,得到所有文档中最相关的k个。 最好的情况是这k个文档在所有机器中平均分布,这时每台机器只要k' = k / n (n为所有机器总数);最坏情况,所有最相关的k个文档只出现在其中的某一台机器上,这时k'需近似等于k了。我觉得比较好的做法可以在每台机器上维护一个堆,然后对堆顶元素实行归并排序。

5. 如第4点所说,对于每个文档d,相对于不同的关键字q1, q2, …, qm,分别有相关性权重f(d, q1),f(d, q2), …, f(d, qm)。如果用户输入关键字qi之后,我们已经获得了最相关的k个文档,而已知关键字qj跟关键字qi相似,文档跟这两个关键字的权重大小比较靠近,那么关键字qi的最相关的k个文档,对寻找qj最相关的k个文档有没有帮助呢?
解答:肯定是有帮助的。在搜索关键字qj最相关的k个文档时,可以在qj的“近义词”相关文档中搜索部分,然后在全局的所有文档中再搜索部分。

寻找第(前)k大的数的方法总结相关推荐

  1. c++求区间第k大数_寻找第K大的数的方法总结

    今天看算法分析是,看到一个这样的问题,就是在一堆数据中查找到第k个大的值. 名称是:设计一组N个数,确定其中第k个最大值,这是一个选择问题,当然,解决这个问题的方法很多,本人在网上搜索了一番,查找到以 ...

  2. 寻找第k大的数的方法总结

    转自:http://www.cnblogs.com/zhjp11/archive/2010/02/26/1674227.html 今天看算法分析是,看到一个这样的问题,就是在一堆数据中查找到第k个大的 ...

  3. 算法-寻找第K大的数的方法总结

    转载:http://www.cnblogs.com/zhjp11/archive/2010/02/26/1674227.html 解法3: void findKthBigger(vector<i ...

  4. C语言(CED)输出前k大的数(分治法/局部快速排序):给定一个数组,统计前k大的数并且把这k个数从大到小输出。

    )输出前k大的数(分治法/局部快速排序):给定一个数组,统计前k大的数并且把这k个数从大到小输出. [输入] 第一行包含一个整数n,表示数组的大小. 第二行包含n个整数,表示数组的元素,整数之间以一个 ...

  5. 信息学奥赛一本通(1235:输出前k大的数)——堆排序

    1235:输出前k大的数 时间限制: 1000 ms         内存限制: 65536 KB 提交数: 12715     通过数: 4043 [题目描述] 给定一个数组,统计前k大的数并且把这 ...

  6. 输出前k大的数(信息学奥赛一本通-T1235)

    [题目描述] 给定一个数组,统计前k大的数并且把这k个数从大到小输出. [输入] 第一行包含一个整数n,表示数组的大小.n < 100000. 第二行包含n个整数,表示数组的元素,整数之间以一个 ...

  7. (分治)7617:输出前k大的数

    描述  给定一个数组,统计前k大的数并且把这k个数从大到小输出. 输入  第一行包含一个整数n,表示数组的大小.n < 100000.  第二行包含n个整数,表示数组的元素,整数之间以一个空格分 ...

  8. C++ 输出前K大的数

    1:输出前k大的数 查看 提交 统计 提问 总时间限制: 10000ms 单个测试点时间限制: 1000ms 内存限制: 65536kB 描述 给定一个数组,统计前k大的数并且把这k个数从大到小输出. ...

  9. java查找第k大的数字_[经典算法题]寻找数组中第K大的数的方法总结

    今天看算法分析是,看到一个这样的问题,就是在一堆数据中查找到第k个大的值. 名称是:设计一组N个数,确定其中第k个最大值,这是一个选择问题,当然,解决这个问题的方法很多,本人在网上搜索了一番,查找到以 ...

最新文章

  1. Android2.3操作系统即将发布,亮点解读
  2. ApacheBench-web性能测试
  3. 002_Spring Data JPA CRUD
  4. Java Web系列:Spring MVC基础
  5. kangle支不支持PHP_【转载】PHP调用kangle的API
  6. 23.多线程 实现的两种方式
  7. URlRewriter组件下载及使用说明
  8. VA_X 飘云阁过期后的处理办法
  9. html5在线聊天模板,h5聊天室模板|仿微信聊天室html5
  10. ubuntu system setting no everthing
  11. SPSS入门笔记记录
  12. 3306π金牌讲师,甜橙金融(翼支付)高级总监张小虎专访
  13. 华为公司面试新员工的有关计算机网络的题目和答案
  14. ctfshow_密码3
  15. 360cdn能挡住cc攻击_关于本站8.11→8.13遭受大型CC攻击和CDN恶意流量攻击,已向公安部报警...
  16. 必知之vs2019添加外加库文件操作
  17. 数组转化为集合的方式asList()
  18. 【MAX78000基础案例演示】
  19. greenplum数据库单机部署
  20. CentOS7.3部署OpenStack-Ocata版本手记(计算节点) - 未完待续

热门文章

  1. 如何修改视频MD5的格式?这些方法值得你收藏
  2. 通过JS方式实现隐藏手机号码中间4位数
  3. 解决linux 2.5G网卡驱动问题
  4. Oracle表查询优化思路
  5. CT影像中肺结节自动检测算法的验证、比较和组合:The LUNA16 challenge
  6. hp固态硬盘安装系统,找不到驱动器和iaStorAfs.sys问题处理
  7. 数学建模番外篇8:画图配色
  8. 手机麦克风结构原理图_驻极体话筒构造和原理
  9. 笔记本电池常识和THINKPAD电源管理器介绍--能设置充电起点和终点
  10. JSP开发模式(四种模式)