在N个数中查找第K大的数字(Top K问题)

article/2025/9/30 15:09:07

在N个乱序数字中查找第k大的数字,时间复杂度可以减小至 

  • O(N*logN)
  • O(N)
  • O(1)
  • O(2)

答案:B

 

所谓“第(前)k大数问题”指的是在长度为n(n>=k)的乱序数组中S找出从大到小顺序的第(前)k个数的问题。

注意:题中只需得到最大的K个数,而不需要对后面N-K个数排序 

可能存在的条件限制:

要求 时间 和 空间消耗最小、海量数据、待排序的数据可能是浮点数等

 

方法一:对所有元素进行排序,之后取出前K个元素,不提倡使用

思路:使用最快排序算法,选择快排 或 堆排

时间复杂度:O(n*logn) + O(K) = O(n*logn)

特点:需要对全部元素进行排序,K = 1 时,时间复杂度也为O(n*logn)

 

方法二:只需要对前K个元素排序,不需要对N-K个元素进行排序,不提倡使用

思路:使用 选择排序 或 起泡排序,进行K次选择,可得到第k大的数

时间复杂度:O(n*k)

 

方法三:不对前K个数进行排序 + 不对N-k个数排序,可以使用

思路:寻找第K个大元素。

具体方法:使用类似快速排序,执行一次快速排序后,每次只选择一部分继续执行快速排序,直到找到第K个大元素为止,此时这个元素在数组位置后面的元素即所求

时间复杂度:

       若随机选取枢纽,线性期望时间O(N)

       若选取数组的“中位数的中位数”作为枢纽,最坏情况下的时间复杂度O(N)

      利用快速排序的思想,从数组S中随机找出一个元素X,把数组分为两部分Sa和Sb。Sa中的元素大于等于X,Sb中元素小于X。这时有两种情况:

           1. Sa中元素的个数小于k,则Sb中的第k-|Sa|个元素即为第k大数;

           2. Sa中元素的个数大于等于k,则返回Sa中的第k大数。

          利用快排的partion思想 T(n) = 2T(n/2) + O(1)   时间复杂度为O(n)   

          该方法只有当我们可以修改输入的数组时可用,位于数组左边的k个数字就是最小的k个数字(但这k个数字不一定是排序的),位于第k个数右边的数字都比第k个数字大

//这里实现的是解法3
#include<iostream>
#include<stdio.h>
using namespace std;int Partition (int *L, int low, int high)
{int temp = L[low];int pt   = L[low]; //哨兵while (low != high){while (low < high && L[high] >= pt)high--;L[low] = L[high];		while (low < high && L[low] <= pt)low++;L[high] = L[low];}	L[low] = temp;return low;
}void QSort (int *L, int low, int high)  //快速排序
{int pl;if (low < high){pl = Partition (L,low,high);QSort (L, low,  pl-1);QSort (L, pl+1, high);}
}void findk(int k,int *L,int low,int high)
{int temp;temp=Partition(L,low,high);if(temp==k-1){cout<<"第"<<temp+1<<"大的数是:"<<L[temp]<<endl;}else if(temp>k-1)findk(k,L,low,temp-1);elsefindk(k,L,temp+1,high);
}int main()
{int a[10]={15,25,9,48,36,100,58,99,126,5},i,j,k;cout<<"排序前:"<<endl;for(i=0;i<10;i++){cout<<a[i]<<" ";}cout<<endl;cout<<"请输入你要查找第k大的数:"<<endl;cin>>k;findk(k,a,0,9); //查找第k大的数不需要全部排序QSort(a,0,9);	cout<<"排序后:"<<endl;for(i=0;i<10;i++){cout<<a[i]<<" ";}cout<<endl;system("Pause");return 0;
}

方法四、我们寻找线性查找的算法,适合数据量小的数据

思路1:寻找第K个大的元素 + 计数排序 + 数组实现

具体方法:使用计数排序,另开辟一个数组,记录每个整数出现的次数,然后再从大到小取最大的 K 个。

缺点:

1、有些数没有出现过,仍要为其保留一个空间,空间浪费比较严重

2、不能处理浮点数

思路2:寻找第K个大的元素 + 计数排序 + map实现

具体方法:利用STL最后的map保存每一个元素Si出现的次数,之后从大到小扫描找到K个数

时间复杂度O(n*logn)     空间复杂度O(n)

注意:

1、可以处理浮点数 

2、不能使用CMap实现,因为Cmap不能根据key自动为其排序

3、map内部是由红黑树实现的,每次插入都是logn,总的复杂度为n*logn。

这里给出两个另外的思路,他们没有计数排序 和 类快速排序好,这里仅仅为了打开思路

 

方法五、基数排序,不提倡使用

思路:寻找第K个大的元素 + 基数排序

一次遍历,找到最大的数为Vmax;,最小的数为Vmin
对区间[Vmin,Vmax]分成M块
 每个小区间的跨度为d=(Vmax–Vmin)/M
 即 [Vmin,Vmin+d], [Vmin+d,Vmin+ 2d],…… 
扫描一遍所有元素,统计各个小区间中的元素个数,我们可以知道第K大的元素在哪一个小区间。
然后,再对那个小区间,继续进行分块 处理。
。。。。递归下去,一直找到一个区间只含第K个数为止

时间复杂度:O ( (N +M )* log2 M (|V max - V min |/delta) )

 

方法六、类二分查找,不提倡使用

思路:寻找第K个大的元素 + 类二分查找

    二分[Smin,Smax]查找结果X,统计X在数组中出现,且整个数组中比X大的数目为k-1的数即为第k大数。时间复杂度平均情况为O(n*logn)

while(Vmax – Vmin > delta)
{Vmid = Vmin + (Vmax - Vmin) * 0.5;if(f(arr,N,Vmid) >= K)Vmin = Vmid;elseVmax = Vmid;
}
伪码中f(arr ,N,Vmid)返回数组arr [0, …, N-1]中大于等于Vmid的数的个数。

举例


结果分析:程序运行的结果,得到一个区间(Vmin, Vmax),这个区间仅包含一个元素(或者多个相等的元素)这个元素就是第K大的元素。

注意:

1、delta的取值要比任意两个不相等的元素差值之最小值小。如果所有元素都是整数,delta可以取值0.5。

2、算法的时间复杂度O( N * log2 (|Vmax - Vmin| /delta) ) - 不知道怎么算的

 

方法七、我们要尽可能少的遍历所有数据。相比下,属于较好的算法,提倡使用

思路:维护一个大小为k的小根堆,堆顶元素是最大K 个数中最小的一个,即第K个元素

处理过程对于数组中的每一个元素X,判断与堆顶的大小
 如果X 比堆顶小,则不需要改变原来的堆, 因为这个元素比最大的K 个 数小。
 如果X比堆顶大,要用X 替换堆顶的元素Y 。调整堆的时间复杂度为O(log2K)。

时间复杂度: O (N * log2 K ),算法只需要扫描所有的数据一次

空间复杂度:大小为K的数组,只需要存储一个容量为K 的堆。

注意、大多数情况下,堆可以全部载入内存。如果K 仍然很大,我们可以尝试先找最大的K ’个元素,然后找第K ’+1个到第2 * K ’ 

元素,如此类推(其中容量K ’的堆可以完全载入内存)。这时,每求出K’个数,就遍历一遍数据了

 

方法八、可以直接对原数组建立大根堆,取这个优先队列前k个值。数据量小的时候可以考虑

思路:在线性时间内,能将一个无序的数组建成一个最小堆,然后取堆中的前k个数

建堆时间是O(n),每次调整时间为O(log n)

复杂度O(n)+k*O(log n)

在有优化,每次调整时不需要调整logn次了,只需调整K次,这个k 和 取第k个数是同一个数

也就是,建堆后,直接取出第一个最大值。取第一个最大值后,大根堆已经被破坏了,之后需要向下进行k次调整就好。取第2个最大值后,之后进行k-1次调整,等等。注意,每次取完值后,这个堆就不是大根堆了

原来堆的方法,每次调整l最大是logn次,调整后仍是大根堆

优化后的时间复杂度是O(n+k^2)

评价:这两个方法的时间复杂度都比维护一个大小为k的小根堆的方法好,但是后者是空间复杂度还是很好的,内存中只需维护一个大小为k堆,而其他两个方法需要把整个堆都放入内存,这对于处理海量数据效率还是不是很好啊,而且作者July还在程序验证过,其实这两种算法在时间上区别不是很大。


http://chatgpt.dhexx.cn/article/gsMwyf2S.shtml

相关文章

idea控制台乱码解决方法

一、问题情况&#xff1a; IntelliJ IDEA 控制台输出中文乱码部分如图所示&#xff1a; 二、解决方法&#xff1a; 1.打开tomcat配置页面&#xff0c;Edit Configurations。 2.选择项目部署的tomcat&#xff0c;在配置项VM options文本框中输入-Dfile.encodingUTF-8,点击Appl…

idea 控制台乱码问题的问题

工程(环境window10&#xff0c;gradle-6.8.3版本&#xff0c;编辑器IntelliJ IDEA 2019.3.4&#xff09; 一&#xff09;问题表现编译build的时候控制台带中文信息的会乱码 1、 找到Help->Editor Custom VM Options 2、 添加 -Dfile.encodingUTF-8 该设置是针对编辑器级别…

IDEA中控制台乱码的解决方式

1.在设置中的“文件编码”中将3个位置设为UTF-8&#xff0c;注&#xff1a;此处设置与控制台乱码无关&#xff0c;3处可均设为UTF-8或均设为系统默认值。 2.在Tomcat的“编辑配置”中&#xff0c;将VM options设为-Dfile.encodingGBK(与第三步类似&#xff0c;默认即为GBK)。…

IDEA控制台乱码终极解决办法

【关于IDEA中文乱码的终极解决方法】 查了很多资料&#xff0c;大多数博主都是修改四个地方&#xff1a;&#xff08;前四步&#xff09;如果前四步还是不行&#xff0c;可以看五、六步。第一步&#xff1a;idea 安装目录下/bin/idea.exe.vmoptions 和/bin/idea64.exe.vmoption…

IDEA控制台乱码解决

解决办法&#xff1a; 打开Intellij的安装的bin目录&#xff08;D:\Program Files\JetBrains\IntelliJ IDEA 14.0\bin &#xff09;&#xff0c;找到上图的两个文件&#xff08;根据你的系统是32位或64位选择其中一个配置文件&#xff09;&#xff0c;在配置文件中添加&#xf…

多种方法帮你解决tomcat项目部署,idea控制台乱码问题

解决在使用Tomcat过程中idea控制台出现的乱码问题 以下将介绍几种方法&#xff08;都是小编亲测实用的方法&#xff09;&#xff0c;尝试并寻找适合自己的方法即可 由于我已经处理过了乱码问题&#xff0c;我就重新配置一下 &#xff08;我有效解决的方案是把-Dfile.encoding…

idea 控制台乱码

1.我原来的修改idea的控制台乱码是 在ideasettings--Editor--File encodings修改为UTF-8 2.tomcat 中修改的conf logging.properties配置文件 由UTF-8修改为GBK 以上是我之前的修改.最近有个项目这样配置之后在项目运行的时候.有个页面要获取一个json字段.但是代码中有乱码的…

IDEA控制台乱码问题,原因解决方式,解决不了算我输

IDEA 控制台乱码问题 文章目录 IDEA 控制台乱码问题为了节省大家时间, 直接展示下我的编码配置方案我的编码配置原则我的编码配置 另外说一下几个重要但是和乱码无关的配置乱码原因解决方式我为什么推荐控制台使用 GBK我的编码设置思想第一种解决方案的弊端end 附加技巧如何找出…

解决idea控制台乱码

控制台乱码如下&#xff1a; 解决方案一&#xff1a;修改当前 Web 项目 Tomcat Server 的虚拟机输出选项 上方导航栏“Run→Edit Configurations…”进入配置页面&#xff0c;修改当前 Web 项目 Tomcat Server 的虚拟机输出选项 VM options 添加 -Dfile.encodingUTF-8 。重启&…

IDEA控制台乱码(已解决)

先来说说我遇到的问题&#xff0c;用IDEA打开项目首先可以保证编辑器内不会乱码&#xff0c;启动Tomcat后控制台出现乱码   我在网上找了很多方式都没有解决&#xff0c;大多数的方式由以下几种&#xff1a; 进入File->Settings->Editor->File Encodings 将右侧的所…

详细解决tomcat乱码 IDEA控制台乱码

1、启动Tomcat时打印出一大堆看不懂的文字 如下图&#xff1a; 原因&#xff1a;产生乱码的根本原因就是编码和解码不一致 解决办法&#xff1a;将打开tomcat的安装目录conf下的logging.properties文件&#xff0c;将java.util.logging.ConsoleHandler.encoding UTF-8 修改为…

MyBatis、IDEA控制台乱码

控制台乱码&#xff1a; 解决方法 在Maven模块的pom.xml中添加 <properties><maven.compiler.source>8</maven.compiler.source><maven.compiler.target>8</maven.compiler.target><project.build.sourceEncoding>UTF-8</project.buil…

【IDEA乱码解决方案】IDEA控制台乱码解决方案收集

乱码实例 如图&#xff1a; 我的解决方案 在“帮助”——“编辑自定义VM选项”中&#xff0c;最后一行加上代码&#xff1a; -Dfile.encodingUTF-8然后重启IDEA。 其他方案一&#xff1a;修改IDEA安装目录下的idea64.exe.vmoptions IDEA快捷方式右键->属性->打开文件所…

idea控制台乱码问题

1.控制台乱码 控制台tomcat启动信息乱码解决&#xff08;红色字体&#xff09; 1 在本地 tomcat 的配置文件中找到 logging.properties 文件设置日志输出的编码为 UTF-8 追加的配置信息为&#xff1a; java.util.logging.ConsoleHandler.encoding UTF-8 2 在IDEA中配置tomcat的…

解决IDEA控制台中文乱码问题(Tomcat、动态网页项目)

博主在使用idea的创建动态网页的时候&#xff0c;遇到了控制台中文乱码问题&#xff0c;在网上参考了多种解决方案之后&#xff0c;终于将问题成功解决。现在将自己遇到问题的情况和解决问题的方法总结如下&#xff1a; Idea控制台中文乱码问题通常有以下两种情况&#xff08;…

idea控制台中文乱码的解决方法(最后一种亲测有效)

idea控制台中文乱码的解决方法&#xff08;三种&#xff0c;亲测有效&#xff09; 问题情况&#xff1a; IntelliJ IDEA 控制台输出中文乱码部分如图所示&#xff1a; 解决方法&#xff1a; 方法一&#xff1a; 1.打开tomcat配置页面&#xff0c;Edit Configurations。 2…

解决IDEA控制台输出中文乱码问题

解决方法一&#xff1a; 1.打开tomcat配置页面&#xff0c;Edit Configurations。 2.在配置项VM options文本框中输入-Dfile.encodingUTF-8&#xff0c; 添加一条JAVA_TOOL_OPTIONS&#xff0c; 点击Apply&#xff0c;OK即可。 3.尝试重启Tomcat。 解决方法二&#xff1a…

解决 IDEA 控制台中文乱码(三种方法)

1、中文乱码原因 IDEA的下方log输出的部分的编码是GBK的&#xff0c;而Tomcat默认log输出是UTF-8编码的&#xff0c;采用了两种不同的编码方式就是乱码 2、Tomcat乱码解决 2-1&#xff09; 右键打开IDEA文件位置&#xff0c;打开下图选中文件 为其添加下图选中代码 -Dfile.e…

IDEA 控制台乱码 解决方法

IDEA 如果不进行配置的话&#xff0c;运行程序时控制台就会中文乱码&#xff0c;严重影响我们对信息的观察 非常的痛苦&#xff0c;那么上解决方法 一.先把idea关掉然后再他的配置文件中改它的编码信息 每个版本的 idea的配置文件可能会有所不同&#xff0c;但不影响 在后面加上…

IDEA 4种解决控制台中文乱码问题

前言 IntelliJ IDEA 如果不进行配置的话&#xff0c;运行程序时控制台中文乱码问题会非常严重&#xff0c;严重影响我们对信息的获取和程序的跟踪。我总结以下 4 点用于解决控制台中文乱码问题&#xff0c;希望有助于大家。 注意&#xff1a;下面根据我日常工作的经验总结&am…