码迷,mamicode.com
首页 > 其他好文 > 详细

BZOJ 1639: [Usaco2007 Mar]Monthly Expense 月度开支

时间:2014-09-10 15:27:30      阅读:187      评论:0      收藏:0      [点我收藏+]

标签:des   style   blog   color   io   ar   for   div   sp   

Description

Farmer John是一个令人惊讶的会计学天才,他已经明白了他可能会花光他的钱,这些钱本来是要维持农场每个月的正常运转的。他已经计算了他以后N(1<=N<=100,000)个工作日中每一天的花费moneyi(1<=moneyi<=10,000),他想要为他连续的M(1<=M<=N)个被叫做“清算月”的结帐时期做一个预算,每一个“清算月”包含一个工作日或更多连续的工作日,每一个工作日都仅被包含在一个“清算月”当中。 FJ的目标是安排这些“清算月”,使得每个清算月的花费中最大的那个花费达到最小,从而来决定他的月度支出限制。

Input

第一行:两个用空格隔开的整数:N和M

第2..N+1行:第i+1行包含FJ在他的第i个工作日的花费

Output

第一行:能够维持每个月农场正常运转的钱数

题解:

M<=N,如果分成<M个清算月满足要求,那么也可以分成M个清算月。

最大值最小,考虑二分。

二分一个ans。

然后模拟一下每一天尽量把钱用完,

得到一个需要的清算月数目cnt。

若cnt<=M合法,否则不合法。

代码:

#include<cstdio>
#include<cstring>
#include<algorithm>
//by zrt
//problem:
using namespace std;
typedef long long LL;
LL a[100005],n,k;
LL maxx=0;
LL l,r;
bool judge(LL x){
    LL sum=0;
    LL cnt=0;
    for(int i=1;i<=n;i++){
        if(sum+a[i]>x){
            sum=a[i];
            cnt++;
            if(cnt>k) return 0;
        }else{
            sum+=a[i];
        }
    }
    if(sum)cnt++;
    if(cnt<=k) return 1;
    else return 0;
}
LL sum;
int main(){
    #ifdef LOCAL
    freopen("in.txt","r",stdin);
    freopen("out.txt","w",stdout);
    #endif
    scanf("%lld%lld",&n,&k);
    for(int i=1;i<=n;i++){
        scanf("%lld",&a[i]);
        maxx=max(maxx,a[i]);
        sum+=a[i];
    }
    l=maxx-1,r=sum;
    while(r-l>1){
        int m=(l+r)>>1;
        if(judge(m)){
            r=m;
        }else l=m;
    }
    printf("%lld\n",r);
    return 0;
}

BZOJ 1639: [Usaco2007 Mar]Monthly Expense 月度开支

标签:des   style   blog   color   io   ar   for   div   sp   

原文地址:http://www.cnblogs.com/zrts/p/bzoj1639.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!