Description
高考结束后,同学们大都找到了一份临时工作,渴望挣得一些零用钱。从今天起,Matrix67将连续工作N天(1<=N<=100 000)。每一天末他可以领取当天及前面若干天里没有领取的工资,但他总共只有M(1<=M<=N)次领取工资的机会。Matrix67已经知道了在接下来的这N天里每一天他可以赚多少钱。为了避免自己滥用零花钱,他希望知道如何安排领取工资的时间才能使得领到工资最多的那一次工资数额最小。注意Matrix67必须恰好领工资M次,且需要将所有的工资全部领走(即最后一天末需要领一次工资)。
Input
第一行输入两个用空格隔开的正整数N和M
以下N行每行一个不超过10000正整数,依次表示每一天的薪水。
以下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 领取到的工资
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; }