python查找第k大的数_寻找数组中第K大的数
给定一个数组A,要求找到数组A中第K大的数字。对于这个问题,解决方案有不少,此处我只给出三种:
方法1:
对数组A进行排序,然后遍历一遍就可以找到第K大的数字。该方法的时间复杂度为O(N*logN)
方法2:
利用简单选择排序法的思想,每次通过比较选出最大的数字来,比较上K次就能找出第K大的数字来。该方法的时间复杂度为O(N*K),最坏情况下为O(N^2)。
方法3:
这种方法是本文谈论的重点,可以利用快排的思想,首先快排每次执行都能确定一个元素的最终的位置,如果这个位置是n-k(其中n是数组A的长度)的话,那么就相当于找到了第K大的元素。设确定的元素位置m的话,如果m > n - k大的话,那么第K大的数字一定
在A[0]~A[m - 1]之间;如果m < n - k的话,那么第K大的数字一定在A[m+1]~A[n - 1]之间。整个过程可以通过递归实现,具体代码如下:
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
using namespace std;
int Partition(int* arr,int low ,int high)
{
int temp = arr[low];
while(low < high)
{
while(low < high && arr[high] >= temp)
high--;
arr[low] = arr[high];
while(low < high && arr[low] <= temp)
low++;
arr[high] = arr[low];
}
arr[low] = temp;//确定参考元素的位置
return low;
}
int KthElement(int * arr,int low, int high,int n ,int k)
{
if(arr == nullptr || low >= high || k > n)//边界条件和特殊输入的处理
return 0;
int pos = Partition(arr,low,high);
while(pos != n - k)
{
if(pos > n - k)
{
high = pos - 1;
pos = Partition(arr,low,high);
}
if(pos < n - k)
{
low = pos + 1;
pos = Partition(arr,low,high);
}
}
return arr[pos];
}
int main()
{
int a[]={1,5,5,7,88,11};
cout<
}
注意:
1.第K大的数字在数组中对应的位置为n-k(按照升序排序的话)。
2.该算法的时间复杂度整体上为O(N)。
3.需要注意的是:这种方法会改变数组中元素的顺序,即会改变数组本身。
4.如果要求第K小的数字的话,只需把n-k换成k-1即可(升序排序)。
===================================================================================================================================================================================================
接下来,我们仔细分析一下方法3的时间复杂度,其实方法3在《算法导论》第九章有着比较详细的描述,但《算法导论》说的是期望为线性时间的选择算法,即该算法的时间复杂度在平均情况下或者一般情况下为O(n);因为此处利用的快排的思想,而快排的时间
复杂度在一般情况下为O(N*logN),但在最坏的情况下(即整个数组原本就是有序的情况)时间复杂度为O(N^2)。所以说对于方法3,《算导》最后给定结果是这样的:平均时间复杂度为O(N),最坏情况下的时间复杂度为O(N^2)。
但是,此处的“平均”同快排一样,是适用于绝大数的情况的。所以我们通常说该算法的时间复杂度为O(N)。
1.我们要搞清楚一点,快排是对参考元素两边都进行递归,而我们的方法3只考虑参考元素的一边,即只对一边进行递归。
2.我们可以粗略的估计下(具体计算还是参考《算导》),在一般情况下方法3的时间复杂度计算公式,假设我们的数据足够的随机,每次划分都在数据序列的中间位置,根据条件1,那么第一次划分我们需要遍历约n个数,第二次需要遍历约n/2个数,...,这样递归下去,最后:
当m趋于无穷大时,该式子收敛于2n,故可以认为其期望时间复杂度为O(N).
===================================================================================================================================================================================================
实际上,《算导》还给出了一种最坏情况下时间复杂度为O(N)的解法,这种方法与方法3类似,都采用了类似快排的思想,但做了一些改变,同时把参考元素也作为输入参数,具体的话此处不再详述,感兴趣可参考《算导》第九章第三节。
python查找第k大的数_寻找数组中第K大的数相关推荐
- java查找第k大的数字_[经典算法题]寻找数组中第K大的数的方法总结
今天看算法分析是,看到一个这样的问题,就是在一堆数据中查找到第k个大的值. 名称是:设计一组N个数,确定其中第k个最大值,这是一个选择问题,当然,解决这个问题的方法很多,本人在网上搜索了一番,查找到以 ...
- 寻找数组中第k大的数
题目:有一个整数数组,请你根据快速排序的思路,找出数组中第K大的数. 给定一个整数数组a,同时给定它的大小n和要找的K(K在1到n之间),请返回第K大的数,保证答案存在. 测试样例: [1,3,5,2 ...
- python第k序列元素查找_【python】寻找数组中第k大的元素
题目链接:https://www.nowcoder.com/question/next?pid=13956292&qid=298692&tid=26431616 方法一: 最简单直接的 ...
- python n个list如何组成矩阵_有序矩阵中第K小的元素amp;x的平方根(二分法篇)
69. x的平方根 题目描述: 实现 int sqrt(int x) 函数. 计算并返回 x 的平方根,其中 x 是非负整数. 由于返回类型是整数,结果只保留整数的部分,小数部分将被舍去. 示例1: ...
- 怎么修改数组中指定元素_求数组中第K大的元素
问题描述 求无序数组int[] nums中第K大的元素. 例如 输入:nums[] = {9,5,8},k = 2 输出:8 输入:nums[] = {3,1,2,4,5,5,6},k = 4 输出: ...
- java寻找数组中第k大的数
快速排序思想,先找数组第k小的数,当执行一次partition函数找到index下标时,index左边的数比arr[index]小,右边的数比arr[index]大 public class Test ...
- Leetcode215 寻找数组中第K大的数
题目描述 给定一个数组A,要求找到数组A中第K大的数字. 思路:对于这种题目,其实有多种解法 方法1:对数组A进行排序,然后遍历一遍就可以找到第K大的数字.该方法的时间复杂度为O(N*logN) 方法 ...
- 寻找数组中第K大的数,时间复杂度O(N)
给定一个数组A,要求找到数组A中第K大的数字.对于这个问题,解决方案有不少,此处我只给出三种: 有道题目寻找第K大,我用这3种方法都做了一遍. 方法1: 对数组A进行排序,然后遍历一遍就可以找到第K大 ...
- java 寻找数组中第k大的值
转载请注明来自:黄朝辉的博客 0.前言 在未排序的数组中查找第k大的数.这里不对k的值进行判断了,认为它是合法的. 1.排序 public static int findKthLargest(int[ ...
最新文章
- idea集成spring+spring MVC+mybatis问题
- js时间戳格式化成日期格式
- Java面试中,一些常见的有关多线程问题!
- 【工厂模式】设计模式之工厂模式【原创】
- hdu 1806线段树 区间合并
- android qml 菜单,QML - ListView项目,用于显示菜单
- Unity搭建简单的图片服务器
- FPGA外挂DDR存储器简介
- 使用Selenium定位鼠标悬浮出现的下拉菜单
- 用Perl编写Apache模块续 - SVNAuth
- 网页音乐视频不让下载怎么办
- HTML5物流大数据服务平台后台模板
- java html转pdf 无法支持中文_java转pdf(html转为pdf),解决中文乱码,标签不规范等问题...
- 淦ORB-SLAM2源码 09--SIM(3)算法
- iPhone, iPad, 的Safari书签和阅读列表不同步问题
- python实现企业微信定时发送文本消息
- Web负载均衡的几种实现方式
- python文本格式上一日_一日一技:在 Python 中快速遍历文件
- 实验四-1:输入一个字符,请判断是字母、数字还是其它字符。
- 天涯明月刀电视剧全集(1-40)