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