`

寻找第K大的数的方法总结

阅读更多
寻找第K大的数的方法总结      

今天看算法分析是,看到一个这样的问题,就是在一堆数据中查找到第k个大的值。

      名称是:设计一组N个数,确定其中第k个最大值,这是一个选择问题,当然,解决这个问题的方法很多,本人在网上搜索了一番,查找到以下的方式,决定很好,推荐给大家。

      所谓“第(前)k大数问题”指的是在长度为n(n>=k)的乱序数组中S找出从大到小顺序的第(前)k个数的问题。
7种解法可供参考(其中最后用空间复杂度换时间效率为O(n)):     
      解法1: 我们可以对这个乱序数组按照从大到小先行排序,然后取出前k大,总的时间复杂度为O(n*logn + k)。
      解法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在数组中出现,且整个数组中比X大的数目为k-1的数即为第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)
(直接用桶排序就好,可以不用hash)

      附注:
      1. STL中可以用nth_element求得类似的第n大的数(由谓词决定),使用的是解法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个数

    接着在这个范围内进行二分查找,找到使得不小于该值的数的数量刚好为K的数值p,即第K大的数。 - **适用场景**:适用于各种规模的数据集。 - **算法复杂度**:时间复杂度为O(N*log(Vmax - Vmin)),空间复杂度低。 - *...

    寻找第k小的数.zip

    总结来说,"寻找第k小的数"这个问题通过分治策略和快速选择算法来解决,可以高效地在未排序的数组中找到特定位置的元素。这种算法设计思想对于理解和优化数据处理程序非常关键,特别是在大数据和实时查询的场景下。

    K尾相等数

    这种方法不仅适用于K尾相等数的问题,还可以推广应用于其他类似问题中,如寻找特定数列中的周期性模式等。此外,通过合理的数据结构(如记录数组)和算法设计,可以在时间和空间上达到较好的性能。

    快速排序输出第k小的数

    在IT领域,尤其是在算法与数据结构的学习和应用中,快速排序是一种非常重要的排序算法,它不仅效率高,而且可以应用于多种场景,包括寻找数组中的第k小元素。根据给定的代码片段,我们可以深入探讨如何利用快速排序...

    基于K近邻的手写数字识别.docx

    基于K-近邻算法的手写数字识别方法在MNIST数据集上的实验结果表明,该方法能够有效地识别手写数字,具有较高的准确率。尽管如此,仍需进一步优化算法参数和增加训练数据量,以提高识别性能。 ##### 6.2 展望 未来的...

    k-means.rar_k means聚类_k-means_k-means方法_k_means matlab_聚类 K MATL

    总结,K-means聚类算法作为一种基础的无监督学习方法,虽然存在一定的局限性,但其简单易用和高效性使其在实际应用中仍然广泛。结合MATLAB的工具支持,我们可以方便地对各种数据进行聚类分析,为后续的数据挖掘和...

    JAVA中寻找最大的K个数解法

    总结来说,寻找最大的K个数可以通过完全排序数组(如快速排序后取前K)或部分排序(如选择排序只找到前K个最大值)来实现。选择排序在某些情况下更有效,因为它不需要对整个数组进行排序,特别是当K远小于数组长度时...

    n位正整数去掉k位数字得到最小数(回溯法C++代码)

    1. **定义问题**:给定一个n位正整数和一个正整数k,目标是从这个n位正整数中删除k个数字,使得剩下的数字组成最小的可能的数。 2. **设计算法**: - 使用一个函数`strdel`来实现字符串中指定位置的字符删除操作。...

    KNN(k近邻算法)最最最全面总结

    1. 计算复杂度高,当数据集较大时,寻找K个最近邻的过程耗时较长。 2. 对于大规模数据集,内存消耗大,因为需要存储所有训练样本。 3. 受到异常值影响较大,一个离群点可能会显著影响结果。 4. 需要预先确定K值,K值...

    奇异值确定K_K值VMD_K._vmd_VMD奇异值_VMDK

    总结来说,利用SVD确定VMD的K值是通过对奇异值分布的分析,找出信号主要成分与噪声之间的界限,从而选择一个合适的分解模式数量。这种方法有助于提取信号的关键特征,提高VMD分解的效率和准确性。

    K-means算法 数据挖掘 k-mean 实验 英文

    K-means算法的核心思想是通过迭代寻找数据的最佳划分,将数据集分为K个簇(cluster),每个簇的中心(centroid)由簇内所有数据点的均值计算得出。这个过程不断重复,直到簇的分配不再发生变化或者达到预设的迭代...

    数值分析几种迭代方法总结

    ### 数值分析中的几种迭代方法总结 #### 一、引言 在数值分析领域,解决线性方程组是常见的任务之一。对于大型稀疏矩阵,直接法往往效率低下且占用大量内存,因此迭代法成为了首选。本文将详细介绍三种常用的迭代...

    大数据量,海量数据 处理方法总结.pdf

    错误率E与位数组大小m、元素数量n和哈希函数个数k的关系是k≈ln2 * (m/n),m≈n * log(1/E) * log_e(log(1/E))。例如,如果误报率为0.01,大约需要m是n的13倍,k约为8。 应用场景:在给定两个文件A和B,每个包含50...

    KNN实现手写数字识别

    KNN算法是一种非参数的监督学习方法,其核心思想是通过寻找训练集中与待分类样本最接近的K个邻居,根据这K个邻居的类别进行投票,选择出现次数最多的类别作为待分类样本的预测类别。KNN算法的关键在于距离度量和K值...

    大数据量,海量数据_处理方法总结

    ### 大数据量、海量数据处理方法总结 #### 一、引言 随着互联网技术的发展,数据量呈现出爆炸性增长的趋势。如何高效地处理这些大数据成为了一项挑战性的任务。在IT行业,尤其是在搜索引擎、社交媒体等领域,处理...

    基于K-means聚类算法的图像分割(MATLAB实现).rar_ASV聚类_K-Means图像分割_k means_rest

    K-means聚类算法是一种广泛应用的数据分析方法,它通过寻找数据集中的“中心点”(或称为质心)来对数据进行分组。在图像处理中,K-means可以用来进行图像分割,将像素划分为不同的类别,以揭示图像的潜在结构。 ...

    机器学习数字字母训练集

    在分类问题中,它通过寻找测试样本最近的K个邻居,并根据这些邻居的类别信息决定测试样本的类别。这个训练集包含了大量的数字和字母样本,可以提供丰富的实例,帮助我们训练出更准确的KNN模型。在实际应用中,KNN...

Global site tag (gtag.js) - Google Analytics