题目描述

输入n个整数,找出其中最小的K个数。例如输入4,5,1,6,2,7,3,8这8个数字,则最小的4个数字是1,2,3,4,。
 
思路:
解法1:对于小规模数据,可以采用类似前题的快速排序思路,pivot之前的点都是比它小的,之后的点都是比它大的。不管是找中位数还是找前k大,前k小,都可以使用这个方法,平均复杂度是O(N),但是最坏时间复杂度是O(N*N)。这样得到最后k个数是没有进行排序的,所以降低了时间复杂度。
这里需要注意idx初值不该设为0,因为找最小的0个数的时候idx = lo = 0;就不会进入循环当中,所以应该注意39行和50行。这题有个大坑就是k的讨论,k可能小于等于0,也可能大于size,所以必须要进行讨论。
这种方法会对输入的数组进行改变,所以应该要明确告诉面试官这样是否允许。而且这种做法适合小规模数据。
 
 class Solution {
public:
int partition(vector<int> &input,int lo,int hi){
int pos = lo + rand() % (hi - lo + );
int pivot = input[pos];
swap(input[lo],input[pos]);
while(lo < hi){
while(lo < hi){
if(input[hi] > pivot){
--hi;
}
else{
input[lo] = input[hi];
++lo;
break;
}
}
while(lo < hi){
if(input[lo] < pivot){
++lo;
}
else{
input[hi] = input[lo];
--hi;
break;
}
}
}
input[lo] = pivot;
return lo;
}
vector<int> GetLeastNumbers_Solution(vector<int> input, int k) {
if(input.empty() || k > input.size() || k <= ){
return {};
}
vector<int> result;
int lo = ,hi = input.size() - ;
int idx ;
idx = partition(input,lo,hi);//
int target = k - ;
while(idx != target){ if(idx < target){
lo = idx + ; }
else if(idx > target){
hi = idx - ;
}
idx = partition(input,lo,hi);
}
for(int i = ;i < k;++i){
result.push_back(input[i]);
}
return result;
}
};

第二种方法:

解法2:O(nlogk)的算法,特别适合处理海量数据。

priority_queue加入和删除操作都是O(logk),top操作是O(1)。默认top的是最大元素。优先级队列本质是基于vector实现的最大堆,因为vector通过下标访问数组,所以top操作复杂度是O(1)的;

STL的set和map还有就是multise和multimap都是基于红黑树实现的,begin是最小元素,即基于最小堆实现的,只有自己写一个仿函数就可以得到最大堆,注意就重载()就行了,stack和queue都是基于deque。红黑树中查找、删除和插入操作的时间复杂度都是O(logk)。

如果要使用仿函数要引入头文件functional。才可以使用greater<int>。

但它是如何保证一棵n个结点的红黑树的高度始终保持在logn的呢?这就引出了红黑树的5个性质:

  1. 每个结点要么是红的要么是黑的。
  2. 根结点是黑的。
  3. 每个叶结点(叶结点即指树尾端NIL指针或NULL结点)都是黑的。
  4. 如果一个结点是红的,那么它的两个儿子都是黑的。
  5. 对于任意结点而言,其到叶结点树尾端NIL指针的每条路径都包含相同数目的黑结点。
#include<iostream>
#include<vector>
#include<string>
#include <cstdio>
#include<queue>
#include<algorithm>
#include<set>
using namespace std;
template<typename T>
struct greater{
bool operator() (const T &a, const T &b) {
return a > b;
}
}; int main() {
multiset<int,greater<int>> hashSet;
for (int i = ; i < ; ++i) {//输入1,3,55,4
int a = ;
cin >> a;
hashSet.insert(a);
}
multiset<int>::iterator iter = hashSet.begin();//得到55
cout << *iter << endl;
system("pause");
}

仿函数最大set

#include<iostream>
#include<vector>
#include<set>
#include<functional>
using namespace std;
int main() {
multiset<int,greater<int>> hashSet;
for (int i = ; i < ; ++i) {//输入1,3,55,4
int a = ;
cin >> a;
hashSet.insert(a);
}
multiset<int>::iterator iter = hashSet.begin();//得到55
cout << *iter << endl;
system("pause");
}

使用functional

class Solution {
public:
vector<int> GetLeastNumbers_Solution(vector<int> input, int k) {
vector<int> result;
if(input.empty() || k <= || k > input.size()){
return result;
}
multiset<int,greater<int>> hashSet;
multiset<int,greater<int>>::iterator iter = hashSet.begin();
for(int i = ;i < input.size();++i){
iter = hashSet.begin();
if(hashSet.size() < k){
hashSet.insert(input[i]);
}
else{
if(*iter > input[i]){
hashSet.erase(*iter);
hashSet.insert(input[i]);
}
}
}
iter = hashSet.begin();
for(iter;iter != hashSet.end();++iter){
result.push_back(*iter);
}
return result;
}
};

解法3:

使用priority_queue实现最大堆。(默认是最大堆,即top等于最大的元素,降序);

底层容器是vector实现的最大堆(stl源码)

模板:priority_queue<int,vector<int>,greater<int>>,升序,最小堆

class Solution {
public:
vector<int> GetLeastNumbers_Solution(vector<int> input, int k) {
vector<int> result;
if(input.empty() || k <= || k > input.size()){
return result;
}
priority_queue<int> q;
for(int i = ;i < input.size();++i){
if(q.size() < k){
q.push(input[i]);
}
else{
if(q.top() > input[i]){
q.pop();
q.push(input[i]);
}
}
}
for(int i = ;i < k;++i){
result.push_back(q.top());
q.pop();
}
return result;
}
};

28 最小的K个数的更多相关文章

  1. 【面试题030】最小的k个数

    [面试题030]最小的k个数 题目:     输入n个整数,找出其中最小的k个数.     例如输入4.5.1.6.2.7.3.8这8个字,则其中最小的4个数字是1.2.3.4.     思路一:   ...

  2. 29.最小的K个数

    题目描述:   输入n个整数,找出其中最小的K个数.例如输入4,5,1,6,2,7,3,8这8个数字,则最小的4个数字是1,2,3,4,. 思路分析:   利用快速排序的partition函数,par ...

  3. 剑指Offer面试题:27.最小的k个数

    一.题目:最小的k个数 题目:输入n个整数,找出其中最小的k个数.例如输入4.5.1.6.2.7.3.8这8个数字,则最小的4个数字是1.2.3.4. 这道题是典型的TopK问题,其最简单的思路莫过于 ...

  4. 算法练习:寻找最小的k个数

    参考July的文章:http://blog.csdn.net/v_JULY_v/article/details/6370650 寻找最小的k个数题目描述:查找最小的k个元素题目:输入n个整数,输出其中 ...

  5. 剑指Offer:面试题30——最小的k个数(java实现)

    问题描述: 输入n个整数,找出其中最小的k个数 思路1: 先排序,再取前k个 时间复杂度O(nlogn) 下面给出快排序的代码(基于下面Partition函数的方法) public void Quic ...

  6. 输入一个数组,求最小的K个数

    被这道题困了好久,看了剑指Offer才知道OJ上的要求有点迷惑性. 题目: 输入n个整数,找出其中最小的K个数.例如输入4,5,1,6,2,7,3,8这8个数字,则最小的4个数字是1,2,3,4. 一 ...

  7. 1046: 最小的K个数

    1046: 最小的K个数 时间限制: 1 Sec  内存限制: 128 MB提交: 233  解决: 200[提交][状态][讨论版] 题目描述 输入n个整数,找出其中最小的K个数.例如输入4,5,1 ...

  8. 最小的K个数:用快排的思想去解相关问题

    实现快速排序算法的关键在于先在数组中选择一个数字,接下来把数组中的数字分为两部分,比选择的数字小的数字移到数组的左边,比选择的数字大的数字移到数组的右边. 这个函数可以如下实现: int Partit ...

  9. 剑指offer面试题30:最小的k个数

    一.题目描述 输入n个整数,找出其中最小的K个数.例如输入4,5,1,6,2,7,3,8这8个数字,则最小的4个数字是1,2,3,4,. 二.解题思路 1.思路1 首先对数组进行排序,然后取出前k个数 ...

随机推荐

  1. 【知识学习】PHP实现批量替换字典后缀

    <?php //要打开字典的物理路径 $filename = 'E:\Local Test\WWW\password.txt'; $handle = fopen($filename,'r') o ...

  2. 【代码审计】XDCMS 报错注入

    审计的都是之前很老的一些的CMS,把学习的过程分享出来,如果有正在和我一起学习的兄弟们,希望看到文章之后会有所收获 ------------------------------------------ ...

  3. 吴裕雄 PYTHON 神经网络——TENSORFLOW 正则化

    import tensorflow as tf import matplotlib.pyplot as plt import numpy as np data = [] label = [] np.r ...

  4. 获取天气预报java代码

    import java.io.BufferedReader; import java.io.FileNotFoundException; import java.io.IOException; imp ...

  5. java里判断字符串是否为数字类型的方法

    String type = "数字类型";if(StringUtils.isNotBlank(value)){ //区分正负数 if(value.startsWith(" ...

  6. 并行效果&迭代器

    具体文章:ALEX的文章 https://www.cnblogs.com/alex3714/articles/5765046.html 串行的并行效果: import time def consume ...

  7. Nginx 反向代理报400错误解决方法!

    如果后端真是的服务器设置有类似防盗链或者根据http请求头中的host字段来进行路由或判断功能的话,如果反向代理层的nginx不重写请求头中的host字段,将会导致请求失败,报400错误,解决办法: ...

  8. FastDFS文件上传和下载流程

    文件上传流程 客户端上传文件后存储服务器将文件 ID 返回给客户端,此文件 ID 用于以后访问该文件的索引信息.文件索引信息包括:组名,虚拟磁盘路径,数据两级目录,文件名.  组名:文件上传后所在的 ...

  9. selenium+chrome options

    selenium+chrome options 环境:selenium chrome 1.      selenium + chrome参数配置 1.1.    启动 from selenium im ...

  10. 【转载】script命令使用

    二.script命令简介当你在终端或控制台上工作时,你可能想记录下自己做了些什么.这种记录可以看成是保存了终端痕迹的文档.假设你跟一些Linux管理员同时在系统上干活.或者说你让别人远程到你的服务器. ...