zoukankan      html  css  js  c++  java
  • XJOI 3601 技能(贪心+二分)

    题目描述:

    有一个oier,他有n个算法技能,每个技能有一个水平值,每个技能的水平上限都是A,设这个oier有cnt个技能达到了A, 设所有水平值的最小值为mi,那么这个oier的战斗力为cnt×Cf+mi×Cm,现在这个oier准备去提升自己的技能,他有m次提升的机会,每次提升可以选择某一个技能给水平值加1,如何分配这些提升的机会,来使得自己的战斗力总值最大?

    输入格式:

    第一行输入5个整数,n,A,Cf,Cm,m

    第二行输入n个整数,表示每个技能的水平值

    输出格式:

    输出最大战斗力值

    样例输入:

    3 5 10 1 5
    1 3 1

    样例输出:

    12

    约定:

    1n100000,1A10^9,0Cf,Cm1000,0m10^15

     

    题解:考虑贪心的策略,由于只有将一个技能点到A或者提升最低技能等级才能增加战斗力,所以说,只有两种操作是有意义的

    1.将一个技能点到A

    2.将所有最低等级的技能都点高一级,相当于提升最低等级一级

    所以我们可以从1-n枚举将i个技能点到a级所需要的最小花费,这个东西sort一下就能搞了,显然点满最大的前i个花费最小,然后用剩下的技能点高最低技能,其思路就相当于把剩下的水倒进低洼,去获得此时水的高度一样

    这个高度显然是可以二分的

    用lower_bound搞出高度大于等于high的第一个位置pos,然后显然pos-1位置都是要填的,至于要填多少,显然可以维护一个前缀和记录已经填了多少了,再用high*(pos-1)-这个数就可以知道你要填多少,然后和你还剩的数比下哪个大就知道了,这东西是单调的,所以可以二分

    不过要注意如果你已经把i个数点到A了,那么pos的最大位置也就是n-i了

    其次开局已经到A的技能还要特殊处理一下

    代码如下:

    #include<deque>
    #include<cstdio>
    #include<cstring>
    #include<iostream>
    #include<algorithm>
    #define mod 1000000007
    using namespace std;
      
    long long a[100010],n,aa,cf,cm,m,sum[100010],tot;
      
    int check(long long high,long long remain,int now)
    {
        if(high>aa) return 0;
        int pos=lower_bound(a+1,a+n+1,high)-a;
        if(pos>n-now) pos=n-now+1;
        if(remain>=high*(pos-1)-sum[pos-1])
        {
            return 1;
        }
        return 0;
    }
      
    int main()
    {
        scanf("%lld%lld%lld%lld%lld",&n,&aa,&cf,&cm,&m);
        for(int i=1; i<=n; i++)
        {
            scanf("%lld",&a[i]);
        }
        sort(a+1,a+n+1);
        while(a[n]==aa)
        {
            tot++;
            n--;
        }
        for(int i=1;i<=n;i++)
        {
            sum[i]=sum[i-1]+a[i];
        }
        long long cnt=0,ans=0;
        long long l=1,r=1e9,mid,re=m-cnt;
        while(l<=r)
        {
            mid=(l+r)>>1;
            if(check(mid,re,0))
            {
                l=mid;
            }
            else
            {
                r=mid-1;
            }
            if(r-l<=1)
            {
                r=check(r,re,0)?r:l;
                break;
            }
        }
        ans=r*1ll*cm+tot*1ll*cf;
        for(int i=1; i<=n; i++)
        {
            cnt+=aa-a[n-i+1];
            if(cnt>=m)
            {
                break;
            }
            long long l=1,r=1e9,mid,re=m-cnt;
            while(l<=r)
            {
                mid=(l+r)>>1;
                if(check(mid,re,i))
                {
                    l=mid;
                }
                else
                {
                    r=mid-1;
                }
                if(r-l<=1)
                {
                    r=check(r,re,i)?r:l;
                    break;
                }
            }
            ans=max(1ll*(i+tot)*cf+1ll*r*cm,ans);
        }
        printf("%lld
    ",ans);
    }

     

     

    思路其实很简单,但是真的是细节很多啊QAQ

     

     

     

  • 相关阅读:
    【转】【VS2008无法启动asp.net development server】的解决
    C#运用技巧(1)
    C# — WinForm 基本控件
    TB 需求分析
    C# 远程连接SQL 2005数据库
    SQL语句的运用
    如何跌倒
    国学堂-梁冬对话张长琳《人体的彩虹》系列
    帝范:中国最伟大帝王的沉思录
    web.xml 配置
  • 原文地址:https://www.cnblogs.com/stxy-ferryman/p/9308509.html
Copyright © 2011-2022 走看看