洛谷 1182 二分 最小化最大值

https://www.luogu.org/problemnew/show/P1182

题目描述

对于给定的一个长度为N的正整数数列A−i,现要将其分成M(M≤N)段,并要求每段连续,且每段和的最大值最小。

关于最大值最小:

例如一数列42451要分成3段

将其如下分段:

[42][45][1]

第一段和为6,第2段和为9,第3段和为1,和最大值为9。

将其如下分段:

[4][24][51]

第一段和为4,第2段和为6,第3段和为6,和最大值为6。

并且无论如何分段,最大值不会小于6。

所以可以得到要将数列42451要分成3段,每段和的最大值最小为6。

输入输出格式

输入格式:

 

第1行包含两个正整数N,M。

第2行包含N个空格隔开的非负整数Ai​,含义如题目所述。

 

输出格式:

 

一个正整数,即每段和最大值最小为多少。

 

输入输出样例

输入样例#1: 复制

5 3
4 2 4 5 1

输出样例#1: 复制

6

说明

对于20%的数据,有N≤10;

对于40%的数据,有N≤1000;

对于100%的数据,有N≤100000,M≤N,Ai​之和不超过10^9。

思路:二分,假设当前分段和最大值为mid,然后看能分多少段,根据结果与m的关系来修改上下限,不懂的推荐看这个专题:

https://blog.csdn.net/xiji333/article/details/86565885

以及这道题让我知道,随便取下限值是会WA的,比如你令l直接取0就会WA掉,虽然我也不知道为什么。

#include<iostream>
#include<algorithm>
#include<cstdio>
#define INF 0x3f3f3f3f
typedef long long ll;
using namespace std;
                    //最小化最大值问题
int a[100005];
int n,m;

int main()
{
    int l=0,r=0;
    scanf("%d%d",&n,&m);
    for(int i=0;i<n;i++)
    {
        scanf("%d",&a[i]);
        r+=a[i];
        l=max(l,a[i]);
    }
    while(l<=r)
    {
        int mid=(l+r)/2;
        int sum=0,cnt=1;
        for(int i=0;i<n;i++)
        {
            sum+=a[i];
            if(sum>mid)
            {
                sum=a[i];
                ++cnt;
            }
            if(cnt>m)
                break;
        }
        if(cnt<=m)   //每段和太大了
            r=mid-1;
        else    //不符合题意 修改下限(最小化问题)
            l=mid+1;
    }
    printf("%d\n",l);
    return 0;
}

 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值