这是一个经典的算法题,下面给出的算法都在给定的数组基础上进行,好处时不用分配新的空间,坏处是会破坏原有的数组,可以自己分配新的空间以避免对原有数组的破坏。

思路一

先直接排序,再取排序后数据的前k个数。

排序算法用最快的堆排序,复杂度也会达到O(N*logN).

void filterDown(int* disorder, int pos, int size){
int temppos=pos,temp=;
while(temppos<size/){
if(*temppos+<size){
if(disorder[*temppos+]>disorder[*temppos+]){
if(disorder[temppos]<disorder[*temppos+]){
temp=disorder[temppos];
disorder[temppos]=disorder[*temppos+];
disorder[*temppos+]=temp;
temppos=*temppos+;
}
else{
break;
}
}
else{
if(disorder[temppos]<disorder[*temppos+]){
temp=disorder[temppos];
disorder[temppos]=disorder[*temppos+];
disorder[*temppos+]=temp;
temppos=*temppos+;
}
else{
break;
}
}
}
else if(disorder[temppos]<disorder[*temppos+]){
temp=disorder[temppos];
disorder[temppos]=disorder[*temppos+];
disorder[*temppos+]=temp;
temppos=*temppos+;
}
else{
break;
} }
}
void heapSort(int* disorder, int size){
int bottomRowSize=;
while(bottomRowSize<size){
bottomRowSize*=;
}
int temp=,i=;
for(int j=size/-;j>=;j--){
filterDown(disorder, j, size);
}
for(int j=size-;j>;j--){
temp=disorder[];
disorder[]=disorder[j];
disorder[j]=temp;
filterDown(disorder,,j);
}
}
int _tmain(int argc, _TCHAR* argv[])
{
const int size=;
const int maxnum = ;
const int k=;
int* disorder=new int[size];
srand((unsigned) time(NULL));
for(int i=;i<size;i++){
disorder[i]=rand()%maxnum;
}
heapSort(disorder,size);
for(int i=;i<k;i++){
cout<<disorder[i]<<endl;
}
return ;
}

当k接近于N时,可以用这种算法。

思路二

先排序前k个数,对于后面N-k个数,依次进行插入。

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

void insertSort(int* disorder, int size){
int temp=,i=;
for(int j=;j<size;j++){
temp=disorder[j];
for(i=j;i>;i--){
if(temp<disorder[i-]){
disorder[i] = disorder[i-];
}
else{
break;
}
}
disorder[i]=temp;
}
} int _tmain(int argc, _TCHAR* argv[])
{
const int size=;
const int maxnum = ;
const int k=;
int* disorder=new int[size];
srand((unsigned) time(NULL));
int i=,temp;
for(;i<size;i++){
disorder[i]=rand()%maxnum;
}
insertSort(disorder,k);
for(int j=k;j<size;j++){
temp=disorder[j];
for(i=k-;i>=;i--){
if(temp<disorder[i]){
disorder[i+] = disorder[i];
}
else{
break;
}
}
disorder[i+]=temp;
} for(int i=;i<k;i++){
cout<<disorder[i]<<endl;
}
return ;
}

当k很小时,可以用这种算法

思路三

对前k个数,建立最大堆,对于后面N-k个数,依次和最大堆的最大数比较,如果小于最大数,则替换最大数,并重新建立最大堆。

时间复杂度为O(N*logk)

void filterDown(int* disorder, int pos, int size){
int temppos=pos,temp=;
while(temppos<size/){
if(*temppos+<size){
if(disorder[*temppos+]>disorder[*temppos+]){
if(disorder[temppos]<disorder[*temppos+]){
temp=disorder[temppos];
disorder[temppos]=disorder[*temppos+];
disorder[*temppos+]=temp;
temppos=*temppos+;
}
else{
break;
}
}
else{
if(disorder[temppos]<disorder[*temppos+]){
temp=disorder[temppos];
disorder[temppos]=disorder[*temppos+];
disorder[*temppos+]=temp;
temppos=*temppos+;
}
else{
break;
}
}
}
else if(disorder[temppos]<disorder[*temppos+]){
temp=disorder[temppos];
disorder[temppos]=disorder[*temppos+];
disorder[*temppos+]=temp;
temppos=*temppos+;
}
else{
break;
} }
} int _tmain(int argc, _TCHAR* argv[])
{
const int size=;
const int maxnum = ;
const int k=;
int* disorder=new int[size];
srand((unsigned) time(NULL));
int i=,temp;
for(;i<size;i++){
disorder[i]=rand()%maxnum;
}
for(int j=k/-;j>=;j--){
filterDown(disorder, j, k);
}
for(int j=k;j<size;j++){
if(disorder[j]<disorder[]){
disorder[]=disorder[j];
filterDown(disorder,,k);
}
}
for(int j=k-;j>;j--){
temp=disorder[];
disorder[]=disorder[j];
disorder[j]=temp;
filterDown(disorder,,j);
}
for(int i=;i<k;i++){
cout<<disorder[i]<<endl;
}
return ;
}

当k和N都很大时,这种算法比前两种算法要快很多。

找出N个数中最小的k个数问题(复杂度O(N*logk))的更多相关文章

  1. Java找N个数中最小的K个数,PriorityQueue和Arrays.sort()两种实现方法

    最近看到了 java.util.PriorityQueue.刚看到还没什么感觉,今天突然发现他可以用来找N个数中最小的K个数. 假设有如下 10 个整数. 5 2 0 1 4 8 6 9 7 3 怎么 ...

  2. 求一个数组中最小的K个数

    方法1:先对数组进行排序,然后遍历前K个数,此时时间复杂度为O(nlgn); 方法2:维护一个容量为K的最大堆(<算法导论>第6章),然后从第K+1个元素开始遍历,和堆中的最大元素比较,如 ...

  3. [算法]找到无序数组中最小的K个数

    题目: 给定一个无序的整型数组arr,找到其中最小的k个数. 方法一: 将数组排序,排序后的数组的前k个数就是最小的k个数. 时间复杂度:O(nlogn) 方法二: 时间复杂度:O(nlogk) 维护 ...

  4. LeetCode 5071. 找出所有行中最小公共元素(Java)

    题目:5071. 找出所有行中最小公共元素 给你一个矩阵 mat,其中每一行的元素都已经按 递增 顺序排好了.请你帮忙找出在所有这些行中 最小的公共元素. 如果矩阵中没有这样的公共元素,就请返回 -1 ...

  5. 求给定数据中最小的K个数

    public class MinHeap { /* * * Top K个问题,求给定数据中最小的K个数 * * 最小堆解决:堆顶元素为堆中最大元素 * * * */ private int MAX_D ...

  6. 【算法】数组与矩阵问题——找到无序数组中最小的k个数

    /** * 找到无序数组中最小的k个数 时间复杂度O(Nlogk) * 过程: * 1.一直维护一个有k个数的大根堆,这个堆代表目前选出来的k个最小的数 * 在堆里的k个元素中堆顶的元素是最小的k个数 ...

  7. 《程序员代码面试指南》第八章 数组和矩阵问题 找到无序数组中最小的k 个数

    题目 找到无序数组中最小的k 个数 java代码 package com.lizhouwei.chapter8; /** * @Description: 找到无序数组中最小的k 个数 * @Autho ...

  8. 小米笔试题:无序数组中最小的k个数

    题目描述 链接:https://www.nowcoder.com/questionTerminal/ec2575fb877d41c9a33d9bab2694ba47?source=relative 来 ...

  9. Python3求m以内的素数、求m个数中最小的n个数

    [本文出自天外归云的博客园] 题1:求m以内的素数(m>2) def find_all_primes_in(m): def prime(num): for i in range(2, num): ...

随机推荐

  1. 大话Spring Cloud

    研究了一段时间Spring Boot了准备向Spring Cloud进发,公司架构和项目也全面拥抱了Spring Cloud.在使用了一段时间后发现Spring Cloud从技术架构上降低了对大型系统 ...

  2. python进行机器学习(五)之模型打分

    一.画出模型的残差值分布情况 #!/usr/bin/python import pandas as pd import numpy as np import csv as csv import mat ...

  3. Django 1.10中文文档-第一个应用Part4-表单和通用视图

    本教程接Part3开始.继续网页投票应用程序,并将重点介绍简单的表单处理和精简代码. 一个简单表单 更新一下在上一个教程中编写的投票详细页面的模板polls/detail.html,让它包含一个HTM ...

  4. 关于解决coursera视频缓冲问题

    关于解决coursera视频缓冲问题 之前使用coursera,不FQ的话,视频根本加载不出来,于是每次都FQ过去看的视频.后来发现可以直接修改hosts就可以了. 以下方法来源知乎的回答(侵删). ...

  5. 使用GDB命令行调试器调试C/C++程序【转】

    转自:https://linux.cn/article-4302-1.html 编译自:http://xmodulo.com/gdb-command-line-debugger.html作者: Adr ...

  6. HighGUI图形图像界面初步——鼠标操作

    OpenCV中的鼠标操作和滑动条的消息映射方式很类似,都是通过一个中介函数配合一个回调函数来实现的,创建和指定滑动条回调函数为createTrackbar, 而指定鼠标操作消息回调函数的函数为setM ...

  7. android 与JS之间的交互

    在页面布局很复杂并且是动态的时候,android本身的控件就变得不是那么地灵活了,只有借助于网页的强大布局能力才能实现,但是在操作html页面的同时也需要与android其它的组件存在交互,比如说 在 ...

  8. C#ActiveX安装项目

    C#开发的ActiveX控件发布方式有三种: 制作客户端安装包,分发给客户机安装: 制作在线安装包,客户机联机安装: 使用html中object的codebase指向安装包地址. 以下为制作安装包: ...

  9. django-1366, "Incorrect string value: '\\xE6\\x88\\x9A\\xE4\\xBC\\x9F...'

    今天把之前的一些代码转移到另外一台电脑的时候, python manage.py syncdb 的时候报了 (1366, "Incorrect string value: '\\xE6\\x ...

  10. opencv python实用操作

    画多边形 fillConvexPloy与fillConvexPloy的区别 fillConvexPloy 用来画单个凸多边形: 如果点集的连线不是凹多边形,则会找一个最小的凸多边形把该凹多边形包住画出 ...