输出前 k 大的数

Posted 寄蜉蝣于天地,渺沧海之一粟

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了输出前 k 大的数相关的知识,希望对你有一定的参考价值。

总时间限制: 10000ms 单个测试点时间限制: 1000ms 内存限制: 65536kB
描述

给定一个数组,统计前k大的数并且把这k个数从大到小输出。

输入
第一行包含一个整数n,表示数组的大小。n < 100000。
第二行包含n个整数,表示数组的元素,整数之间以一个空格分开。每个整数的绝对值不超过100000000。
第三行包含一个整数k。k < n。
输出
从大到小输出前k大的数,每个数一行。
样例输入
10
4 5 6 9 8 7 1 2 3 0
5
样例输出
9
8
7
6
5

分析:

按照快速排序的思想,把数组前k大的数放到数组末尾。然后在对数组末尾k个元素做排序再输出该部分元素。

 1 #include<stdio.h>
 2 #include<stdlib.h>
 3 
 4 int a[100010];
 5 
 6 int cmp(const void *a,const void *b)
 7 { return (*(int *)a) - (*(int *)b); }
 8 
 9 //将a[]数组下标区间[start,end]前k大的数放到数组下标在[start,end]范围的末尾部分.
10 void FindMaxK(int a[],int start,int End,int k)
11 {
12     if(start-End+1==k) return;//若是元素个数刚好就是k个,则可以直接返回. 
13     int i=start,j=End,key=a[start];
14     while(i<j)
15     {
16         while(i<j&&a[j]>=key) --j;
17         a[i]=a[j];
18         while(i<j&&a[i]<=key) ++i;
19         a[j]=a[i];
20     }
21     a[i]=key;
22     if(End-i+1==k) return;//数组后半段的元素个数为End-i+1,刚好够k个 
23     else if( End-i+1 > k) FindMaxK(a,i+1,End,k);//数组后半段的元素个数多于k个 
24     else FindMaxK(a,start,i-1,k-(End-i+1) );//数组后半段元素个数不够k个。所以要在前半段继续寻找k-(End-i+1)这么多个。 
25 }
26 int main()
27 {
28     int n,k;
29 
30     scanf("%d",&n);
31     for(int i = 0;i <n; ++i) scanf("%d",&a[i]);
32     scanf("%d",&k);
33     
34     FindMaxK(a,0,n-1,k);
35 
36     qsort(a+n-k,k,sizeof(a[0]),cmp);
37     for(int i = n-1;i >= n-k; --i) printf("%d\\n",a[i]);
38     return 0;
39 }

 

C ++版:(北大郭炜老师)

 1 #include <iostream>
 2 #include <cstdio>
 3 #include <algorithm>
 4 using namespace std;
 5 
 6 int a[100010];
 7 
 8 void swap(int & a,int & b)
 9 {  int tmp = a;  a = b;  b = tmp;  }
10 
11 //将a[]数组下标区间[start,end]前k大的数放到数组下标在[start,end]范围的末尾部分.
12 void FindMaxK(int a[],int start,int End,int k)
13 {
14     if(start-End+1==k) return;//若是元素个数刚好就是k个,则可以直接返回. 
15     int i=start,j=End,key=a[start];
16     while(i<j)
17     {
18         while(i<j&&a[j]>=key) --j;
19         swap(a[i],a[j]);
20         while(i<j&&a[i]<=key) ++i;
21         swap(a[i],a[j]);
22     }
23     if(End-i+1==k) return;//数组后半段的元素个数为End-i+1,刚好够k个 
24     else if( End-i+1 > k) FindMaxK(a,i+1,End,k);//数组后半段的元素个数多于k个 
25     else FindMaxK(a,start,i-1,k-(End-i+1) );//数组后半段元素个数不够k个。所以要在前半段继续寻找k-(End-i+1)这么多个。 
26 }
27 int main()
28 {
29     int n,k;
30     
31     scanf("%d",&n);
32     for(int i = 0;i <n; ++i) scanf("%d",&a[i]);
33     scanf("%d",&k);
34     
35     FindMaxK(a,0,n-1,k);
36 
37     sort(a+n-k-1,a+n);
38     for(int i = n-1;i >= n-k; --i)
39         printf("%d\\n",a[i]);
40     return 0;
41 }

本问题可以参考阅读:

http://www.cnblogs.com/macher/p/5317439.html

http://www.cnblogs.com/huashanqingzhu/p/6591091.html

 

以上是关于输出前 k 大的数的主要内容,如果未能解决你的问题,请参考以下文章

openjudge 7617:输出前k大的数

输出前 k 大的数

OpenJ_Bailian 7617 输出前k大的数

统计前k大的数x

(分治)输出前m大的数。。。

查找无序数组中第K大的数