zoukankan      html  css  js  c++  java
  • 【noip模拟赛6】收入计划 最大值的最小值 二分答案

    描述

     

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

    输入

     

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

    以下N行每行一个不超过10000正整数,依次表示每一天的薪水。

    输出

     

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

    输入样例 1 

    7 5
    100
    400
    300
    100
    500
    101
    400

    输出样例 1

    500

    提示

    【样例说明】

    采取下面的方案可以使每次领到的工资不会多于500。这个答案不能再少了。

    100 400   300 100   500   101   400   每一天的薪水

    <------1 <-------2 <---3 <---4 <---5  领取工资的时间

    500       400     500   101   400   领取到的工资

    老是想着用dp   

    之前做过一次最大值的最小值二分答案  以后一定要引起警觉

    二分答案的时候出现了很多小错误 

    有以下两种写法要注意:

    #include<bits/stdc++.h>
    using namespace std;
    //input
    #define rep(i,a,b) for(int i=(a);i<=(b);i++)
    #define RI(n) scanf("%d",&(n))
    #define RII(n,m) scanf("%d%d",&n,&m);
    #define RIII(n,m,k) scanf("%d%d%d",&n,&m,&k)
    #define RS(s) scanf("%s",s);
    #define LL long long
    #define REP(i,N)  for(int i=0;i<(N);i++)
    #define CLR(A,v)  memset(A,v,sizeof A)
    //////////////////////////////////
    #define N 100005
    int a[N];
    int main()
    {
        int n,m;
        RII(n,m);
        long sum=0;
        int maxx=0;
        rep(i,1,n)
        {
            RI(a[i]);
            sum+=a[i];
            maxx=max(maxx,a[i]);
        }
        long long L=maxx;
        long long R=sum;
       int ans;
        while(L<R)
        {
            long long sum=0;
            long long mid=(L+R)>>1;
    
            int cnt=0;
            rep(i,1,n)
            {
                if(sum+a[i]>mid)
                {
                    cnt++;sum=a[i];
                }
                else
                    sum+=a[i];
            }
            if(sum)cnt++;
    
            if(cnt>m)L=mid+1;
            else  R=mid;
            
            //如果写成这样就会错  因为mid是趋向左的  如果L和M差1 那么将进入死循环
            if(cnt>=m)L=mid;
            else R=mid-1;
        }
        cout<<R;
    }
    View Code
    #include<bits/stdc++.h>
    using namespace std;
    //input
    #define rep(i,a,b) for(int i=(a);i<=(b);i++)
    #define RI(n) scanf("%d",&(n))
    #define RII(n,m) scanf("%d%d",&n,&m);
    #define RIII(n,m,k) scanf("%d%d%d",&n,&m,&k)
    #define RS(s) scanf("%s",s);
    #define LL long long
    #define REP(i,N)  for(int i=0;i<(N);i++)
    #define CLR(A,v)  memset(A,v,sizeof A)
    //////////////////////////////////
    #define N 100005
    int a[N];
    
    int main()
    {
        int n,m;
        RII(n,m);
        long sum=0;
        int maxx=0;
        rep(i,1,n)
        {
            RI(a[i]);
            sum+=a[i];
            maxx=max(maxx,a[i]);
        }
    
        long long L=maxx;
        long long R=sum;
       int ans;
        while(L<=R)
        {
            int cnt=0;
            long long sum=0;
            long long mid=(L+R)>>1;
           
            rep(i,1,n)
            {
                if(sum+a[i]>mid)
                {
                    cnt++;sum=a[i];
                }
                else
                    sum+=a[i];
            }
            if(sum)cnt++;
            
            if(cnt>m)L=mid+1;
            else  ans=mid,R=mid-1;
    
        }
        cout<<ans;
    
    }
    View Code
  • 相关阅读:
    Java中的并发库学习总结
    Android源码的git下载地址
    Android下载Android源码
    工具类 验证手机邮箱
    hibernate缓存说明
    Hibernate之N+1问题
    常用正则
    Java Base64加密、解密原理Java代码
    Base64加密解密原理以及代码实现(VC++)
    情书经典语录
  • 原文地址:https://www.cnblogs.com/bxd123/p/10554820.html
Copyright © 2011-2022 走看看