(纪中)2416. Berry Picking【数学】

119 篇文章 0 订阅

(File IO): input:berries.in output:berries.out
时间限制: 1000 ms 空间限制: 262144 KB 具体限制
Goto ProblemSet


题目描述
B e s s i e Bessie Bessie 和她的妹妹 E l s i e Elsie Elsie 正在 F a r m e r J o h n Farmer John FarmerJohn 的浆果园里采浆果。 F a r m e r J o h n Farmer John FarmerJohn 的浆果园里有 N N N棵浆果树 ( 1 ≤ N ≤ 1000 ) (1≤N≤1000) 1N1000;树 i i i 上有 B i Bi Bi 个浆果 ( 1 ≤ B i ≤ 1000 ) (1≤Bi≤1000) 1Bi1000 B e s s i e Bessie Bessie K K K个篮子( 1 ≤ K ≤ 1000 1≤K≤1000 1K1000 K K K 为偶数)。每个篮子里可以装同一棵树上采下的任意多个浆果,但是不能装来自于不同的树上的浆果,因为它们的口味可能不同。篮子里也可以不装浆果。
B e s s i e Bessie Bessie 想要使得她得到的浆果数量最大。但是, F a r m e r J o h n Farmer John FarmerJohn 希望 B e s s i e Bessie Bessie 与她的妹妹一同分享,所以 B e s s i e Bessie Bessie 必须将浆果数量较多的 K / 2 K/2 K/2 个篮子给 E l s i e Elsie Elsie。这表示 E l s i e Elsie Elsie 很有可能最后比 B e s s i e Bessie Bessie 得到更多的浆果,这十分不公平,然而姐妹之间往往就是这样。
帮助 B e s s i e Bessie Bessie 求出她最多可以得到的浆果数量。


输入
输入的第一行包含空格分隔的整数 N N N K K K
第二行包含 N N N 个空格分隔的整数 B 1 , B 2 , … , B N B1,B2,…,BN B1,B2,,BN

输出
输出一行,包含所求的答案。


样例输入
5 4
3 6 8 4 2

样例输出
8


数据范围限制
测试点 1 − 3 1-3 13 满足 K ≤ 10 K≤10 K10
测试点 4 − 10 4-10 410 没有额外限制。


提示
如果 B e s s i e Bessie Bessie 在一个篮子里装树 2 2 2 6 6 6 个浆果
两个篮子里每个装树 3 3 3 4 4 4 个浆果
一个篮子里装树 4 4 4 4 4 4 个浆果
那么她能够得到两个各装有 4 4 4 个浆果的篮子,总共 8 8 8 个浆果。


解题思路
假设 E l s i e Elsie Elsie 拿的篮子里面果子数最小值为 m m m,那么最好情况是她拿的篮子里全都是 m m m 个果子。我们可以从 1 1 1 m a x max max a i a_i ai枚举这个 m,令能装满的(也就是装了 m m m 个果子的)篮子数为 L L L。分类讨论:

  • L < 2 K L< 2K L<2K:不能满足最小条件,停止枚举。
  • L ≥ K L≥K LK:此时 B e s s i e Bessie Bessie 能拿到的就是 m K / 2 mK/2 mK/2个果子,更新答案即可。
  • 这里的关键是每棵树上 在装满若干篮子之后 剩余的果子数,也就是 a i m o d a_i mod aimod m m m。我们以 a i m o d a_i mod aimod m m m为关键字从大到小对 a 数组排序,排序结果为, B e s s i e Bessie Bessie 拿的果子数就是:
    在这里插入图片描述

代码

#include<iostream>
#include<cstring>
#include<string>
#include<cstdio>
#include<algorithm>
#include<iomanip>
#include<cmath>
using namespace std;
int n,k,a[1010],maxn,f,ans,t,bsy;
bool cmp(int x,int y)
{
    return x%t>y%t;
}
int main(){
	freopen("berries.in","r",stdin);
	freopen("berries.out","w",stdout);
	scanf("%d%d",&n,&k);
    for(int i=1;i<=n;i++)
    {
        scanf("%d",&a[i]);
        maxn=max(maxn,a[i]);
    }
    for(int i=1;i<=maxn;i++)
    {
        f=0;
        for(int j=1;j<=n;j++)
            f+=a[j]/i;
        if(f<(k/2))
            break;
        if(f>=k)
        {
            ans=max(ans,i*(k/2));
            continue;
        }
        t=i;
        sort(a+1,a+n+1,cmp);
        bsy=(f-k/2)*t;
        for(int x=1;x<=n,x+f<=k;x++)
            bsy+=a[x]%t;
        ans=max(ans,bsy);
    }
    printf("%d",ans);
}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值