Description
2015胡润全球财富榜调查显示,个人资产在1000万以上的高净值人群达到200万人,假设给出N个人的个人资产值,请你快速找出排前M位的大富翁。
Input
首先输入两个正整数N( N ≤ 10^6)和M(M ≤ 10),其中N为总人数,M为需要找出的大富翁数目,接下来给出N个人的个人资产,以万元为单位,个人资产数字为正整数,数字间以空格分隔。
Output
一行数据,按降序输出资产排前M位的大富翁的个人资产值,数字间以空格分隔,行末不得有多余空格。
Sample
Input
6 3
12 6 56 23 188 60
Output
188 60 56
Hint
请用堆排序完成。
这道题好像只能用堆排序完成,而且我感觉就是堆排序的模板题,因为他输入的数据量特别大,但是需要输出的却比较少,而且是最大的几个!
这个对排序只用的下沉操作,当然如果需要添加新数据的话,好像需要上浮操作…个人理解…
#include <bits/stdc++.h>
using namespace std;
int n,m;
int a[15];
void Sift_Down(int k)
{
/*
堆的下调函数,给定节点编号k,进行下调
*/
int i=k;
while(i<=m/2)
{
int t;
if(a[2*i]<a[i])
t=2*i;
else
t=i;
if(2*i+1<=m&&a[2*i+1]<a[t])
t=2*i+1;
if(t!=i)
{
a[0]=a[i];//以a[0]作为交换变量
a[i]=a[t];
a[t]=a[0];
i=t;
}
else
{
return;
}
}
}
int main()
{
scanf("%d %d",&n,&m);
for(int i=1;i<=m;i++)
scanf("%d",&a[i]);
for(int i=m/2;i>=1;i--)
Sift_Down(i);
for(int i=m+1;i<=n;i++)
{
int x;
scanf("%d",&x);
if(x>a[1])
{
a[1]=x;
Sift_Down(1);
}
}
int k=m;
for(int i=1;i<k;i++)
{
a[0]=a[1];
a[1]=a[m];
a[m]=a[0];
m--;
Sift_Down(1);
}
for(int i=1;i<k;i++)
printf("%d ",a[i]);
printf("%d\n",a[k]);
return 0;
}