CF985C

Posted repulser

tags:

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

CF985C

题意:

你要组成N个木桶,组成每个木桶需要K个木块,(第二行给你N*K个木块),使得任意两个木桶之间的差值不超过L的情况,使得所有木桶可以装的水的和最大,输出这个最大和,如果无法满足要求输出0。

解法:

因为众所周知的木板原理,我们考虑贪心,对木板的长度进行排序。
易知,让长度相差小的木板组合比较优。
首先考虑没答案的情况,假设最短的木板为a[0],那么所有木桶中能装水的最小值就确定了为a[0]。所以其他木桶的最短板不能超过a[0]+L,如果范围在a[0]~(a[0]+L)的木板少于N块肯定不行
排序upper_bound求一下即可,假设有pos块。
如满足要求,考虑范围在a[0]~(a[0]+L)木块,我们尽量的使短的木板组合在一起,那么就是[0,k)个组成第一个桶(如果剩余木板还够N-1个),前[K,2k)个组成第二个桶(如果剩余木板还够N-2个)...前[iK,(i+1)*k)个组成第i个桶(如果剩余木板还够N-i个)。如果选取了i个之后,剩余不够再这样组合了,我们直接从这些木块中选取最后的 N-i 块即可。

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>

using namespace std;

#define LL long long
#define N 100010

int m,n,k,l,a[N];

int main() 
    scanf("%d%d%d",&n,&k,&l);
    m = n * k;
    for(int i = 0 ; i < m ; i++)
        scanf("%d",&a[i]);
    sort(a,a + m);
    int pos = upper_bound(a,a + m,a[0] + l) - a;
    LL ans = 0;
    if(n <= pos) 
        int i;
        for(i = 0 ; pos - i * k > n - i ; i++) ans += a[i * k];
        for(int j = pos - 1 ; j >= pos - n + i ; j--) ans += a[j];
    
    printf("%lld \n",ans);
//    system("pause");
    return 0;

以上是关于CF985C的主要内容,如果未能解决你的问题,请参考以下文章

codeforces 985C Liebig's Barrels

关于安排

CF 爆发者

[CF930E]/[CF944G]Coins Exhibition

CF怎么改名

dp专题