有若干个互不相等的无序的数,怎么选出其中最大的k个数。

我自己的方案:因为学过找第k大数的O(N)算法,所以第一反应就是找第K大的数。然后把所有大于等于第k大的数取出来。

写这个知道算法的代码都花了2个多小时,反思,太慢了。 注意边界条件,不要混混沌沌的。

/************我自己的解法*****************/
//选择数组a[N]中最大的第k个数
int Select(int * a, int N, int k)
{
if(k > N || a == NULL)
{
cout << "error!!!";
}
if(N < )
{
qsort(a, N, sizeof(a[]), cmp);
return a[N - k];
}
int arrNum = N/; //可以整除5的数字
int surplus = N % ; //除以5剩下的数字
int * mid = new int [arrNum]; //存储每一组的中位数
int * S1 = new int [N];
int * S2 = new int [N];
int posS1 = ;
int posS2 = ; for(int i = ; i < arrNum; i++)
{
qsort(a+i*, , sizeof(a[]), cmp);
mid[i] = *(a + i * + );
} int midNum = Select(mid, arrNum, arrNum/); for(int i = ; i < arrNum; i++)
{
int * currentGroup = a + i * ;
if(currentGroup[] < midNum)
{
memcpy(S1 + posS1, currentGroup, * sizeof(a[]));
posS1 += ;
for(int j = ; j < ; j++)
{
if(currentGroup[j] <= midNum)
S1[posS1++] = currentGroup[j];
else
S2[posS2++] = currentGroup[j];
}
}
else
{
memcpy(S2 + posS2, currentGroup + , * sizeof(a[]));
posS2 += ;
for(int j = ; j < ; j++)
{
if(currentGroup[j] <= midNum)
S1[posS1++] = currentGroup[j];
else
S2[posS2++] = currentGroup[j];
}
} }
for(int i = arrNum * ; i < N; i++)
{
if(a[i] <= midNum)
S1[posS1++] = a[i];
else
S2[posS2++] = a[i];
} if(k == posS2)
{
return midNum;
}
else if(k < posS2)
{
return Select(S2, posS2, k);
}
else
{
return Select(S1, posS1, k - posS2);
} } int * GetBiggestK(int * a, int N, int k)
{
int * biggestK = new int [k];
int numK = Select(a, N, k);
int pos = ;
for(int i = ; i < N; i++)
{
if(a[i]>=numK)
{
biggestK[pos++] = a[i];
}
} return biggestK; }

编程之美对这种算法的评价是:由于这个线性算法的常数项比较大,在实际应用中有时效果并不好。

---------------------------------------------------------------------------------------------------------------------------

书上的解法:

解法一: 先排序,再取出最大的k个数 复杂度 O(NlogN)

解法二:类似于快速排序,先在数据中选一个数做标准,把数据划分为两个部分,大于等于选中数字S1个 和 小于选中数字S2个 。如果S1大于k则在S1个数中继续找最大的k个,如果S1小于k,则把这S1个数加入答案,并在S2中继续找其他的数字。 平均复杂度为O(NlogK).

/*********答案中 解法二*****************/
int * GetBiggestKAns2(int * a, int N, int k)
{
if(a == NULL || k > N)
{
cout << "error!!!"<< endl;
return NULL;
}
int first = a[];
int head = ;
int tail = N - ;
int numAlready = ; int postion = GetPosition(a, first, head, tail);
swap(a[], a[postion]); if(postion + == k)
{
int * ans = new int [k];
memcpy(ans, a, k * sizeof(a[]));
return ans;
}
else if(postion + > k)
{
return GetBiggestKAns2(a, postion + , k);
}
else
{
int * ans = new int [k];
memcpy(ans, a , (postion + ) * sizeof(a[]));
int * ansPart = GetBiggestKAns2(a + postion + , N - postion - , k - postion - );
memcpy(ans + postion + , ansPart, (k - postion - ) * sizeof(a[]));
return ans;
}
} //交换时让大数在前,小数在后
int GetPosition(int * a, int first, int p, int r)
{
int head = p;
int tail = r;
while(head < tail)
{
while(a[head] >= first)
head++;
while(a[tail] < first)
tail--;
if(head < tail)
{
swap(a[head], a[tail]);
head++;
tail--;
}
}
return head - ; //如果所有的数字都比first大 那么head会指向数组后的第一个数字
//如果所有的数字都比first小 那么head会指向第一个数字 都是恰好head - 1 是应该交换的位置
}

----------------------------------------------------------------------------------------------------

解法三:用二分法找第k大的数 数据分布平均情况下,时间复杂度为O(NlogN)

伪代码如下:

while(Vmax - Vmin > delta)
{
Vmid = Vmin + (Vmax - Vmin) * 0.5;
if(f(arr, N, Vmid) >= K)
{
Vmin = Vmid;
}
else
{
Vmax = Vmid;
}
}

f(arr, N, Vmid) 返回数组arr[0...N-1]中大于等于Vmid的数的个数。delta是比数组中数字的最小差值还要小的数。

最后[Vmid, Vmax]只剩下一个数,就是第k大的数。

另一个思路:若所有的数都是整数,可以比较他们的二进制位,从最高位开始,1的是较大的数 有A个,0的是较小的数有B个。若 A > K则在A中找最大的K个,反之 把A个数加入答案, 在B中找 最大的K - A个数。

----------------------------------------------------------------------------------------------------

解法四:用一个最小堆存储最大的那K个数,先把K个数加入最小堆,新来一个数就与最小堆的根节点对比,比根节点小丢弃,比根节点大则替换根节点,调整最小堆。这样遍历完后最小堆里的数字就是最大的K个数。 时间O(NlogK)

-----------------------------------------------------------------------------------------------------

解法五:若数字的范围是有限的整数,可以分配一个count[MAXN]的数组,count[0]记录0在数组中出现的次数...这样只要遍历一遍数组得到count再遍历一遍count就可以找到最大的K个数了。

【编程之美】2.5 寻找最大的k个数的更多相关文章

  1. 编程之美2.5:寻找最大的K个数

    编程之美2.5:寻找最大的K个数 引申:寻找第k大的数: 方法一: // 选择第k大的数(通过改进快速排序来实现) public static void SelectShort(int[] array ...

  2. 算法系列:寻找最大的 K 个数

    Copyright © 1900-2016, NORYES, All Rights Reserved. http://www.cnblogs.com/noryes/ 欢迎转载,请保留此版权声明. -- ...

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

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

  4. 第2章 数字之魅——寻找最大的K个数

    寻找最大的K个数 问题描述 在面试中,有下面的问答: 问:有很多个无序的数,我们姑且假定它们各不相等,怎么选出其中最大的若干个数呢? 答:可以这样写:int array[100] …… 问:好,如果有 ...

  5. O(N)的时间寻找最大的K个数

    (转:http://www.cnblogs.com/luxiaoxun/archive/2012/08/06/2624799.html) 寻找N个数中最大的K个数,本质上就是寻找最大的K个数中最小的那 ...

  6. 03寻找最小的k个数

    题目描述:查找最小的k个元素         题目:输入n个整数,输出其中最小的k个.         例如输入1,2,3,4,5,6,7和8这8个数字,则最小的4个数字为1,2,3和4. 1:最简单 ...

  7. 寻找最大的k个数问题

    这是编程之美书第2.5节的一道题目. 各种解法: 解法一,用nlgn复杂度的排序算法对数组进行从大到小排序,取前K个.但这方法做了两件不必要做的事:它对想得到的K个数进行了排序,对不想得到的n-K个数 ...

  8. 寻找最大的k个数

    这个题目是非常经典的一个题目,解法也有很多,现在就把我已经理解的解法记录下来. 题目描述 有n个无序的数,它们各不相等,怎样选出其中的最大的k个数呢? 题目分析: 解法1: 最容易想到的就是把n个数进 ...

  9. 算法笔记_035:寻找最小的k个数(Java)

    目录 1 问题描述 2 解决方案 2.1 全部排序法 2.2 部分排序法 2.3 用堆代替数组法 2.4线性选择算法   1 问题描述 有n个整数,请找出其中最小的k个数,要求时间复杂度尽可能低. 2 ...

随机推荐

  1. 迷你版Deferred

    直接贴代码: /** * 迷你版的deferred */ function Deferred(func) { if (this instanceof Deferred === false) { ret ...

  2. glusterFS的常用命令 (转)

    1.       启动/关闭/查看glusterd服务 # /etc/init.d/glusterd start # /etc/init.d/glusterd stop # /etc/init.d/g ...

  3. iOS: ARC & MRC下string内存管理策略探究

    ARC & MRC下string内存管理策略探究 前两天跟同事争论一个关于NSString执行copy操作以后是否会发生变化,两个人整了半天,最后写代码验证了一下,发现原来NSString操作 ...

  4. 【bzoj1864】[ZJOI2006]三色二叉树

    题目描述 输入 仅有一行,不超过500000个字符,表示一个二叉树序列. 输出 输出文件也只有一行,包含两个数,依次表示最多和最少有多少个点能够被染成绿色. 样例输入 1122002010 样例输出 ...

  5. Redis学习笔记三:多机数据库的实现

    1.复制 执行slaveof命令或者设置slaveof选项,让一个服务器去复制另外一个服务器. 旧版复制功能的实现(Redis 2.8 之前的版本) 复制功能分为同步和命令传播两个操作. 同步(syn ...

  6. android Home键和返回键

    在Android中,当按下Home键,默认情况下stop前台的actiity,即activity设置成onstop,而不是ondestory.如果再次启动该activity不是调用onCreate,而 ...

  7. android ListView嵌套GridView显示不全问题

    只需重写GridView即可:public class MyGridView extends GridView{ public MyGridView(android.content.Context c ...

  8. UNITY3D在线更新之道-CSlight 使用总结

    转自:http://blog.csdn.net/leonwei/article/details/39233775 最近做U3D的热更新,研究了各种方式无果后,最容易最先想到的方式就是利用c#的反射机制 ...

  9. NOIP“对偶”题:还教室

    先说一下思路: 方差可以经过恒等变形变成 x12 + x22 + ... + xn2 + 2a(x1 + x2 + ... + xn) + na2 所以维护平方和.连续和即可 平均数我就不再推了…… ...

  10. iOS使用webView 加载网页,在模拟器中没有问题,而真机却白屏了。App Transport Security has blocked a cleartext HTTP (http://) resource load since it is insecure. Temporary exceptions can be configured via your app's Info.plist f

    还在info.plist中配置.除了配置允许上网的配置之外,还有另一项.