当前位置:首页 > 题解目录 > 正文内容

2020CSPJ-直播获奖

亿万年的星光3年前 (2022-10-07)题解目录2219

【题目描述】

NOI2130 即将举行。为了增加观赏性,CCF 决定逐一评出每个选手的成绩,并直播即时的获奖分数线。本次竞赛的获奖率为w%,即当前排名前 w% 的选手的最低成绩就是即时的分数线。

更具体地,若当前已评出了p 个选手的成绩,则当前计划获奖人数为 

max(1,pw%),其中 w 是获奖百分比,x 表示对x 向下取整,max(x,y) 表示 x 和y 中较大的数。如有选手成绩相同,则所有成绩并列的选手都能获奖,因此实际获奖人数可能比计划中多。

作为评测组的技术人员,请你帮 CCF 写一个直播程序。

【输入描述】

第一行有两个整数 n,w。分别代表选手总数与获奖率。

第二行有 n 个整数,依次代表逐一评出的选手成绩。

【输出描述】

只有一行,包含 n 个非负整数,依次代表选手成绩逐一评出后,即时的获奖分数线。相邻两个整数间用一个空格分隔。

【样例输入】

10 60
200 300 400 500 600 600 0 300 200 100

【样例输出】

200 300 400 400 400 500 400 400 300 300

【提示】

样例 1 解释:

已评测选手人数12345678910
计划获奖人数1112334456
已评测选手的分
数从高到低排列
(其中,分数线
粗体标出)
200300
200
400
300
200
500
400
300
200
600
500
400
300
200
600
600
500
400
300
200
600
600
500
400
300
200
0
600
600
500
400
300
300
200
0
600
600
500
400
300
300
200
200
0
600
600
500
400
300
300
200
200
100
0


注意,在第9名选手的成绩评出之后,计划获奖人数为5人,但由于有并列,实际会有6人获奖。

输入样例2:

10 30
100 100 600 100 100 100 100 100 100 100

输出样例2:

100 100 600 600 600 600 100 100 100 100

【数据规模与约定】:

各测试点的n 如下表:

测试点编号n=
1∼310
4∼6500
7∼102000
11∼1710^4
18∼2010^5


对于所有测试点,每个选手的成绩均为不超过600 的非负整数,获奖百分比w 是一个正整数且 1≤w99

【提示】

在计算计划获奖人数时,如用浮点类型的变量(如 C/C++ 中的 float 、 double,Pascal 中的 real 、 double 、 extended 等)存储获奖比例w%,则计算5×60% 时的结果可能为3.000001,也可能为 2.999999,向下取整后的结果不确定。因此,建议仅使用整型变量,以计算出准确值。




【题目分析】

  • 排序题,最简单的思路是每次新加一个人,然后就sort一遍,但是这样做肯定会超时的(n=100000)。

  • 其实,从题目中给的测试数据可以发现,每个选手的成绩并不大,不超过600。那么比较好的做法就是用桶排了。


【参考代码1】

得分:85分,有三个测试点超时

#include<bits/stdc++.h>
using namespace std;
int t[605];  //成绩数组
int n,w;  //选手总数,获奖率
int main()
{
    int x; //
    cin>>n>>w;
    for(int i=1;i<=n;i++)  //循环读入
    {
        cin>>x;
        t[x]++;   //桶排,表示得分x出现的次数
        int sum=0;
        for(int j=600;j>=0;j--)  //最大是600,遍历,从高到低
        {
            sum+=t[j]; //累计得分人数
            int num = max(1,i*w/100); //计划获奖人数,取较大的那个
            if(sum>= num) 
            {
                cout<<j<<' ';  //符合条件的输出
                break;
            }
        }
    }
    return 0;
 }


问题原因:cin、cout影响了输入输出效率,改成下面这样用scanf和printf就是满分了。

#include<bits/stdc++.h>
using namespace std;
int t[605];  //成绩数组
int n,w;  //选手总数,获奖率
int main()
{
    int x; 
    scanf("%d%d",&n,&w);
    for(int i=1;i<=n;i++)  //循环读入
    {
        scanf("%d",&x);
        t[x]++;   //桶排,表示得分x出现的次数
        int sum=0;
        for(int j=600;j>=0;j--)  //最大是600,遍历,从高到低
        {
            sum+=t[j]; //累计得分人数
            int num = max(1,i*w/100); //计划获奖人数,取较大的那个
            if(sum>= num) 
            {
                printf("%d ",j);  //符合条件的输出
                break;
            }
        }
    }
    return 0;
 }


扫描二维码推送至手机访问。

版权声明:本文由青少年编程知识记录发布,如需转载请注明出处。

标签: cspj
分享给朋友:

相关文章

【题解】月度开销

【题目描述】农夫约翰是一个精明的会计师。他意识到自己可能没有足够的钱来维持农场的运转了。他计算出并记录下了接下来N(1 ≤N≤ 100,000) 天里每天需要的开销。约翰打算为连续的M(1 ≤M≤N)...

【题解】石子合并

【题目描述】在一个操场上一排地摆放着N堆石子。现要将石子有次序地合并成一堆。规定每次只能选相邻的2堆石子合并成新的一堆,并将新的一堆石子数记为该次合并的得分。设计一个程序,计算出将N堆石子合并成一堆的...

【题解】特殊的质数肋骨

【题目描述】农民约翰母牛总是产生最好的肋骨。你能通过农民约翰和美国农业部标记在每根肋骨上的数字认出它们。农民约翰确定他卖给买方的是真正的质数肋骨,是因为从右边开始切下肋骨,每次还剩下的肋骨上的数字都组...

【题解】切割钢管

【题解】切割钢管

【题目描述】小A是某工地的计算工程师。工地现有 n 根钢管,第 i 根钢管的长度为 ai。现在想用这 n 根钢管来做一个支撑用的柱子。我么可以切割这些钢管成为更短的钢管,但是不能缝合两根钢管。为了安全...

【题解】开关灯(2)

1.开关灯(light.cpp)【题目描述】某实验室共有n盏灯,灯的编号为1~n,每盏灯的初始状态是关闭的。现在有m位学生,每位学生可以前去抽取一张带数字的卡片,其数字为Ai,然后依次将自己手中的数字...

【题解】循环比赛日程表

【题目描述】设有N个选手进行循环比赛,其中 N=2^M ,要求每名选手要与其他的N-1名选手都赛一次,每名选手每天比赛一次,循环赛共进行N-1天,要求每天没有选手轮空。【输入描述】输入M【输出描述】一...