【POJ - 3273】Monthly Expense (二分)

Monthly Expense

直接上中文

Descriptions

给你一个长度为N的序列,现在要让你把他们切割成M份(所以每一份都是连续的),然后每一份都有一个和sum[i],其中最大的一个是maxSum = max(sum[i]),问这个最大值最小是多少?

输入

多组输入输出
每组数据第一行是2个整数N,M(1<=M<=N<=100000),接着是N行,每行一个整数vi,表示这个序列.

输出

每组数据输出一行一个数,为这个最大值最小是多少

输入样例

7 5
100
400
300
100
500
101
400

输出样例

500

样例解释

对于样例,你可以把100,400分成第一组,300,100分成第二组,500分成第三组,101分成第四组,400分成第五组,他们的和最大的是500。

题目链接

https://vjudge.net/problem/POJ-3273

一开始上届是n个数的和。代表只分一组。

下届是n个数中最大的数。代表分n组;

如果结果是mid=(上届+下届)/2;那么根据mid看看能分多少组。组数大于m,代表mid比实际值小。下届变成mid+1。否则,上届变成mid-1;

AC代码

#include <iostream>
#include <cstdio>
#include <fstream>
#include <algorithm>
#include <cmath>
#include <deque>
#include <vector>
#include <queue>
#include <string>
#include <cstring>
#include <map>
#include <stack>
#include <set>
#include <sstream>
#define IOS ios_base::sync_with_stdio(0); cin.tie(0);
#define Mod 1000000007
#define eps 1e-6
#define ll long long
#define INF 0x3f3f3f3f
#define MEM(x,y) memset(x,y,sizeof(x))
#define Maxn 100000+5
using namespace std;
int N,M;
int l,r,mid;//左右中
int a[Maxn];//序列
int solve()
{
    mid=(l+r)/2;//预判的最大值
    int sum=0;//当前这份的和
    int cnt=1;//有cnt份
    for(int i=0; i<N; i++)
    {
        if(sum+a[i]<=mid)//当前这份的和不能大于预判的最大值
            sum+=a[i];
        else
        {
            sum=a[i];
            cnt++;
        }
        if(cnt>M)//所得份数大于题目要求的份数,即预判和小了
            return 0;
    }
    return 1;//所得份数小于等于题目要求的份数,即预判和大了
}
int main()
{
    while(cin>>N>>M)
    {
        l=r=0;//初始化
        for(int i=0; i<N; i++)
        {
            cin>>a[i];
            l=max(l,a[i]);
            r+=a[i];
        }
        while(l<=r)
        {
            if(!solve())//预判和小了,整个取值区间要变大,即左区间向右移动
               l=mid+1;
            else//预判和大了,整个取值区间要变小,即右区间向左移动
                r=mid-1;
        }
        cout<<mid<<endl;
    }
}

猜你喜欢

转载自www.cnblogs.com/sky-stars/p/11299196.html