【Tyvj1359】收入计划

Description

    高考结束后,同学们大都找到了一份临时工作,渴望挣得一些零用钱。从今天起,Matrix67将连续工作N天(1<=N<=100 000)。每一天末他可以领取当天及前面若干天里没有领取的工资,但他总共只有M(1<=M<=N)次领取工资的机会。Matrix67已经知道了在接下来的这N天里每一天他可以赚多少钱。为了避免自己滥用零花钱,他希望知道如何安排领取工资的时间才能使得领到工资最多的那一次工资数额最小。注意Matrix67必须恰好领工资M次,且需要将所有的工资全部领走(即最后一天末需要领一次工资)。

Input

    第一行输入两个用空格隔开的正整数N和M
    以下N行每行一个不超过10000正整数,依次表示每一天的薪水。

Output

   输出领取到的工资的最大值最小是多少。

Sample Input

67 5 
100 
400 
300 
100 
500 
101 
400

Sample Output

500

HINT

    采取下面的方案可以使每次领到的工资不会多于500。这个答案不能再少了。
100 400   300 100   500   101   400   每一天的薪水
<------1 <-------2 <---3 <---4 <---5  领取工资的时间
  500       400     500   101   400   领取到的工资
 
 
/*
  二分答案
  注意左边界是该序列中最大的数 
*/ 

#include<iostream>
#include<cstdio>
using namespace std;
int a[100010];
int n,m,sum=0,ans=0;
int jud(int x){
    int tim=0,pos=0;
    for (int i=1;i<=n;i++) 
        {
            pos+=a[i];
            if (pos>x) pos=a[i],tim++;
        }
    return tim+1<=m;
}
int main(){
    int maxx=0;
    scanf("%d%d",&n,&m);
    for (int i=1;i<=n;i++)
    {
        scanf("%d",&a[i]);
        sum+=a[i];
        if (a[i]>maxx) maxx=a[i];
    }
    int l=maxx,r=sum;
    while (l<=r){
        int mid=(l+r)/2;
        if (jud(mid)) ans=mid,r=mid-1;
            else l=mid+1;
    }
    printf("%d",ans);
    return 0;
}

 

转载于:https://www.cnblogs.com/liumengyue/p/5137747.html

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值