【编程题目】查找最小的 k 个元素
5.查找最小的 k 个元素(数组)
题目:输入 n 个整数,输出其中最小的 k 个。
例如输入 1,2,3,4,5,6,7 和 8 这 8 个数字,则最小的 4 个数字为 1,2,3 和 4。
算法里面学过查找第k小的元素的O(n)算法
试着实现了一下:
注意new 初始化二维数组的方式
int (* a)[5] = new int[8][5];
/*
5.查找最小的 k 个元素(数组)
题目:输入 n 个整数,输出其中最小的 k 个。
例如输入 1,2,3,4,5,6,7 和 8 这 8 个数字,则最小的 4 个数字为 1,2,3 和 4。
*/ #include <iostream>
#include <algorithm>
using namespace std; bool compare(int a, int b) //降序排列
{
return a > b;
} //找到 一共有 n个 元素的 数组S 中 第k小 的数字 整个算法中,对除以5后余下的数字都做了特殊处理
int Select(int * S, int k, int n)
{
if (n < ) //对少于5个的做特殊处理
{
sort(S, S + n);
return S[k - ];
}
int subn = n/ + ((n % == ) ? : );
int subnn = n/;
int (* subS)[] = new int[subn][];
for (int i = ; i < subnn; i++)
{
for (int j = ; j < ; j++)
{
subS[i][j] = S[i * + j];
}
sort(subS[i], subS[i] + , compare); //5个一组,每组从大到小排序
}
for (int j = ; j < n % ; j++)
{
subS[subn - ][j] = S[subn * + j - ];
}
sort(subS[subn - ], subS[subn - ] + n % , compare); int * M = new int [subn];
for (int i = ; i < subn; i++)
{
M[i] = subS[i][]; //M 中存储每组数字的中位数
} int Mn = subnn;
int m = Select(M, Mn/ + (Mn % == ) ? : , Mn);
delete [] M; int * S1 = new int [n]; //存放小于等于m的数字
int * S2 = new int [n];
int S1n = ; //记录有多少小于等于m的数字
int S2n = ; //找到相应的S1 与 S2中的元素
for (int i = ; i < subnn; i++)
{
if (subS[i][] <= m)
{
for (int j = ; j < ; j++)
{
if (j < )
{
if (subS[i][j] <= m)
{
S1[S1n++] = subS[i][j];
}
else
{
S2[S2n++] = subS[i][j];
}
}
else
{
S1[S1n++] = subS[i][j];
}
}
}
else
{
for (int j = ; j < ; j++)
{
if (j > )
{
if (subS[i][j] <= m)
{
S1[S1n++] = subS[i][j];
}
else
{
S2[S2n++] = subS[i][j];
}
}
else
{
S2[S2n++] = subS[i][j];
}
}
}
}
if (subnn != subn) //多余的数字特别处理
{
for (int j = ; j < n % ; j++)
{
if (subS[subn - ][j] > m)
{
S2[S2n++] = subS[subn - ][j];
}
else
{
S1[S1n++] = subS[subn - ][j];
}
}
} if (k == S1n)
{
delete [] S1;
delete [] S2;
return m;
}
else if (k < S1n)
{
return Select(S1, k, S1n);
}
else
{
return Select(S2, k - S1n, S2n);
} delete [] S1;
delete [] S2;
} int main()
{
int a[] = {,,,,,,,};
int m = Select(a, , ); return ;
}
不过,我的代码看起来好长,好难受啊...
网上有用堆做的,对堆不是很了解,要补一下知识。
看了用堆的方法的原理,理论上会比我现在实现的这个算法慢一点
方法是用堆维护k个最小元素
下面来自:https://github.com/julycoding/The-Art-Of-Programming-By-July/blob/master/ebook/zh/02.01.md
解法二
咱们再进一步想想,题目没有要求最小的k个数有序,也没要求最后n-k个数有序。既然如此,就没有必要对所有元素进行排序。这时,咱们想到了用选择或交换排序,即:
1、遍历n个数,把最先遍历到的k个数存入到大小为k的数组中,假设它们即是最小的k个数;
2、对这k个数,利用选择或交换排序找到这k个元素中的最大值kmax(找最大值需要遍历这k个数,时间复杂度为O(k));
3、继续遍历剩余n-k个数。假设每一次遍历到的新的元素的值为x,把x与kmax比较:如果x < kmax ,用x替换kmax,并回到第二步重新找出k个元素的数组中最大元素kmax‘;如果x >= kmax,则继续遍历不更新数组。
每次遍历,更新或不更新数组的所用的时间为O(k)或O(0)。故整趟下来,时间复杂度为n*O(k)=O(n*k)。
解法三
更好的办法是维护容量为k的最大堆,原理跟解法二的方法相似:
- 1、用容量为k的最大堆存储最先遍历到的k个数,同样假设它们即是最小的k个数;
- 2、堆中元素是有序的,令k1<k2<...<kmax(kmax设为最大堆中的最大元素)
- 3、遍历剩余n-k个数。假设每一次遍历到的新的元素的值为x,把x与堆顶元素kmax比较:如果
x < kmax,用x替换kmax,然后更新堆(用时logk);否则不更新堆。
这样下来,总的时间复杂度:O(k+(n-k)*logk)=O(n*logk)。此方法得益于堆中进行查找和更新的时间复杂度均为:O(logk)(若使用解法二:在数组中找出最大元素,时间复杂度:O(k))。
堆的实现代码:来自http://www.cnblogs.com/panweishadow/p/3632639.html
public static void FindKMin(int[] sort, int k)
{
int[] heap = sort;
int rootIndex = k / - ;
while (rootIndex >= )
{
reheap(heap, rootIndex, k - );
rootIndex--;
} for (int i = k, len=heap.Length; i < len; i++)
{
if (heap[i]<heap[])
{
heap[] = heap[i];
reheap(heap, , k - );
}
} Console.WriteLine("The {0} min element =",k);
for (int i = ; i < k; i++)
{
Console.Write(heap[i] + " ");
}
} private static void reheap(int[] heap, int rootIndex, int lastInddex)
{
int orphan = heap[rootIndex];
bool done = false;
int leftIndex = rootIndex * + ;
while (!done && leftIndex <= lastInddex)
{
int largerIndex = leftIndex;
if (leftIndex+ <= lastInddex)
{
int rightIndex = leftIndex + ;
if (heap[rightIndex] > heap[leftIndex])
{
largerIndex = rightIndex;
}
} if (orphan < heap[largerIndex])
{
heap[rootIndex] = heap[largerIndex];
rootIndex = largerIndex;
leftIndex = rootIndex * + ;
}
else
{
done = true;
}
} heap[rootIndex] = orphan;
}
【编程题目】查找最小的 k 个元素的更多相关文章
- 查找最小的k 个元素之C#算法实现
紧接着上一篇微软编程面试100题,这次想解决的是查找最小的K个元素,题目是:输入n 个整数,输出其中最小的k 个.例如输入1,2,3,4,5,6,7 和8 这8 个数字,则最小的4 个数字为1,2,3 ...
- 【Data Structure & Algorithm】 查找最小的k个元素
查找最小的k个元素 题目:输入n个整数,输出其中最小的k个. 例如输入1, 2, 3, 4, 5, 6, 7和8这八个数字,则最小的4个数字为1, 2, 3和4. 分析:这道题最简单的思路是把输入的n ...
- 查找最小的K个元素,使用最大堆。
查找最小的K个元素,使用最大堆,具体代码如下: #define _CRT_SECURE_NO_WARNINGS #include <iostream> using namespace st ...
- 5.查找最小的k个元素[Kmin]
[题目] 输入n个整数,输出其中最小的k个. 例如输入1,2,3,4,5,6,7和8这8个数字,则最小的4个数字为1,2,3和4. [分析] 这道题最简单的思路莫过于把输入的n个整数排序,这样排在最前 ...
- 5.查找最小的k个元素(数组)
题目: 输入n个整数,输出其中最小的k个,例如输入1,2,3,4,5,6,7,8这8个数,则最小的4个是1,2,3,4(输出不要求有序) 解: 利用快速排序的partition,算导上求第k大数的思想 ...
- 查找最小的k个元素
题目:输入n个整数,输出其中最小的k个. 例如输入1,2,3,4,5,6,7和8这8个数字,则最小的4个数字为1,2,3和4. 两种思路,无非就是时间与空间的妥协. 限制空间的时候要对原数组进行排序, ...
- 程序员面试50题(1)—查找最小的k个元素[算法]
题目:输入n个整数,输出其中最小的k个.例如输入1,2,3,4,5,6,7和8这8个数字,则最小的4个数字为1,2,3和4. 分析:这道题最简单的思路莫过于把输入的n个整数排序,这样排在最前面的k个数 ...
- 查找最小的k个元素 【微软面试100题 第五题】
题目要求: 输入n个整数,输出其中最小的k个. 例如:输入1,2,3,4,5,6,7,8这8个数字,则最小的4个数字为1,2,3,4. 参考资料:剑指offer第30题. 题目分析: 解法一: 用快排 ...
- 排序,求几个最值问题,输入n个整数,输出其中最小的k个元素。
看完两个求最大值算法之后的一些感想. 如果想直接看算法的可以跳过.但是我觉得我这些想法还是比较有用的,至少对我将来的算法设计是这样的. 算法的功能越强大,必然意味着速度慢,因为根据丛林法则,那种慢又功 ...
随机推荐
- 密码学初级教程(一)基本概念及DES加密算法
密码技术在网络通信中广泛使用,本节是初步接触密码学技术的笔记. 第1章 加密-解密 破译 明文-密文 密钥 密码算法 对称密码-公钥密码(非对称密码) 单向散列函数-散列值 消息认证码 数字签名 伪随 ...
- [译]Node.js : Building RESTful APIs using Loopback and MySQL
国庆后可能就要使用StrongLoop那套东西来做项目了 原文:http://www.javabeat.net/loopback-mysql/ Loopback是什么? Loopback是一个开源的N ...
- 【AngularJS】—— 7 模块化
AngularJS有几大特性,比如: 1 MVC 2 模块化 3 指令系统 4 双向数据绑定 那么本篇就来看看AngularJS的模块化. 首先先说一下为什么要实现模块化: 1 增加了模块的可重用性 ...
- SQL pivot 基本用法 行列转换 数据透视
SQL通过pivot进行行列转换 数据透视 可直接在sql server 运行 传统操作 和 pivot create table XKCl (name nchar(10) not null, 学科 ...
- su root 和su - root 的区别
su - root is the same as su - just like login as root, then the shell is login shell,which mean i ...
- HDOJ 4750 Count The Pairs
按边长从小到大排序...再逐个加入(就像MST一样)最先联通的点之间最长路径中的最小值就是新加入的边的长.... Count The Pairs Time Limit: 20000/10000 MS ...
- 淘宝(阿里百川)手机客户端开发日记第一篇 android 主框架搭建(一)
android 主框架搭建(一) 1.开发环境:Android Studio 相继点击下一步,直接项目建立完毕(如下图) 图片看的效果如果很小,请放大您的浏览器显示百分比 转载请注明http://w ...
- BZOJ2229—— [Zjoi2011]最小割
0.题目大意:求两点之间的最小割,然后找出其中小于x的数量 1.分析:最小割树水题,上个板子就好 #include <queue> #include <ctime> #incl ...
- [转载]JavaEE学习篇之——JQuery技术详解
原文链接:http://blog.csdn.net/jiangwei0910410003/article/details/32102187 1.简介2.工具3.jQuery对象 1.DOM对象转化成j ...
- vb.net dll创建
创建vb.net的动态链接库 如果你想用用VC来编写vb.net的dll,我想本文不适合. 本文只说vb.net的dll. 何为vb.net的dll?实际上就是一个类库. 很多个类封装成一个库了,这就 ...