loj2141 「SHOI2017」期末考试

我们枚举每一个时间点,使得所有科目的时间都小于等于这个时间点,计算安排老师的代价和学生们的不满意度更新答案。

但是枚举太慢了,可以发现,时间点越早,学生们不满意度越小,安排老师的代价越高。即安排老师的代价、学生们的不满意度和时间点的关系一个是负相关一个是正相关的。

因此,安排老师的代价、学生们的不满意度的和是一个以时间点为 \(x\) 轴的凹函数,三分。

#include <algorithm>
#include <iostream>
#include <cstdio>
using namespace std;
typedef long long ll;
int A, B, n, m, t[100005], b[100005];
ll C, ans=0x3f3f3f3f3f3f3f3f, ove, lwr;
ll chk(int lim){
    ll re=0;
    ove = lwr = 0;
    for(int i=1; i<=m; i++)
        if(b[i]>lim)
            ove += (ll)b[i] - lim;
        else
            lwr += (ll)lim - b[i];
    if(A<=B){
        int tmp=min(lwr, ove);
        lwr -= tmp;
        ove -= tmp;
        re += (ll)tmp * A;
        re += (ll)ove * B;
    }
    else    re = ove * B;
    for(int i=1; i<=n; i++)
        if(lim>t[i])
            re += (ll)(lim - t[i]) * C;
    return re;
}
ll sanfen(){
    int l=0, r=100000, midl, midr, dis;
    while(l<=r){
        dis = (r - l + 1) / 3;
        midl = l + dis;
        midr = l + dis + dis;
        ll re1=chk(midl), re2=chk(midr);
        ans = min(ans, re1);
        ans = min(ans, re2);
        if(re1>re2) l = midl + 1;
        else    r = midr - 1;
    }
    return ans;
}
int main(){
    cin>>A>>B>>C>>n>>m;
    //C too large
    int minn=0x3f3f3f3f;
    for(int i=1; i<=n; i++){
        scanf("%d", &t[i]);
        minn = min(minn, t[i]);
    }
    for(int i=1; i<=m; i++)
        scanf("%d", &b[i]);
    if(C>200000){
        cout<<chk(minn)<<endl;
    }
    else
        cout<<sanfen()<<endl;
    return 0;
}

[\(\mathrm{O}(n)\)]做法](http://blog.csdn.net/sdfzyhx/article/details/70597948)

转载于:https://www.cnblogs.com/poorpool/p/8568062.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值