zoukankan      html  css  js  c++  java
  • BZOJ3874 codevs3361 宅男计划

    AC通道1:http://www.lydsy.com/JudgeOnline/problem.php?id=3874

    AC通道2:http://codevs.cn/problem/3361/

    [题目分析]

      [为什么会做到这道题]

      首先会点进这道题,貌似是打CF一题遇到了贪心题[可是有神犇表示用三分法可以A],然后我就想起三分法似乎还从没有打过,于是找到了很久之前%的一篇J.K的博客。

      

      [介绍一下三分法]

      首先还是介绍一下三分法的作用是什么?...

      如果要找一个凸函数的最优值,比如二次函数,也就是最优值在中间,而且两边依次递减的函数时,我们可以使用三分法来逼近答案。[二分法只能找单调函数]

      传统意义三分法操作过程:每次在线段上取两个三分点,比较两点函数值的大小,然后舍弃小的一边,接着操作。

      正确性怎么证明呢?

      首先你脑洞一个单峰,然后在三分之一处描点,然后不管你怎么画,单峰一定不在较小值到最近端点的那一段。

      如果两点在单峰的异侧,那么删掉任意1/3都保证单峰仍在区间内。

      如果在同侧,那么靠近单峰的点一定>远离单峰的点,删掉小的一边,单峰仍在区间内。

      为什么一定要画两个点呢?

      因为两个点才好比较啊...一个点显然不够,因为不知道单峰在哪边,三个点好像又多了,于是优美的两个点。

      为什么一定要取1/3处呢?

      因为这样每次将线段长度*2/3,复杂度是稳定的。复杂度是log的。不过你设计一个别的也是可以的。

      例如说在冬令营时宋老师提到一个神奇的黄金比例分割式三分。这个的原理是什么呢?

      每次我选择两个点ml,mr,满足这样的性质,将[l,ml]的删去后,mr是新区间的ml;将[mr,r]删去后,ml是新区间的mr。

      然后列一个等比关系,解出来ml=l+(3-根号5)/2*(r-l+1),mr=r-(3-根号5)/2*(r-l+1)。这样的话每次选新区间的时候,就只需要再多算一个点了。

      好机智啊!它的复杂度也差不多,每次是原长*0.618...但实际运用中,浮点误差不可忽略,而且坐标是整点表示,于是这个算法bug较多,屡次80-90分WA,所以考场上采用上面的方法比较靠谱。

      还有同学觉得不靠谱,因为区间大小<3时,三分好像就除不下去了,于是在最后的区间里暴力求一遍也是极好的。

      

      [开始了第一阶段的思考----2.24]

      好吧,我们再回来看这个题。

      如果我告诉你要叫快递小哥t次,你能不能求出最多宅几天呢?...

      仔细思考一下...应该是可以求出来的。首先我们可以排除掉一些不可能点的外卖,比如保质期短还很贵的。如果有保质期比它长或和它一样并且价格便宜的,我就可以舍弃它了。

      现在我就得到了一个真正会购买的序列,满足a[i].s<a[i+1].s,a[i].p<a[i+1].p就是前一个比这一个保质期短,并且价格比这个便宜。

      可以想象,我每次购买应该尽量平均,为什么呢?比如我一次买10天的,结果第二次只买1天的,不如1次买6天,1次买5天[因为保质期越久越贵嘛]。

      那么贪心下来就是,考虑第一个保质期最短但是最便宜的商品,我每次当然都会购买,但是我能买多少呢?当然是要么买到没钱,要么买到能吃到保质期结束那天的个数。

      那么往下接着考虑,保质期第二短的,我买完第一短的当然接着买咯,还是买到没钱,或者就是在第一个保质期过之后到第二个保质期之间都吃它。然后一直这样考虑下去...

      [注意]上面买的时候都是给t组都买一份,但是如果发现买不到这个保质期过那么多个了...那我选择剩下的钱都买,然后放到t个中的某些次数中去。

      

      贪心的部分也搞定了,但是我现在的问题是不知道要叫小哥多少次啊。

      最少0次,最多m/(f+a[1],p)次,复杂度是不能考虑枚举的。

      但是我们发现上面说了一大段的三分法,诶,我们好像可以用三分法咯?

      三分法使用条件:单峰函数...

      宅几天和外卖次数之间为什么会呈现单峰函数呢?是人性的扭曲?还是道德的沦丧?

      笔者也觉得有点奇怪= =,J.K.是这么说的:“我不能确保这种方法的正确性,因为迄今为止我还没有看到其他能够复杂度能够承受的办法,最起码这样做的话,数据是可以过的,当然不排除数据不够全面。因为送物品非常自由,没有任何限制,所以我们要找一个合适的自变量进行枚举。可以发现,如果我们外卖的次数过少,那么就会出现一些食品性价比不高的情况;如果次数过多,那么就会浪费外卖运费。故可以从这里入手,因为可以看出这是一个类似于二次函数的函数。我们可以通过三分来查找峰值。

      说的好啊,笔者后来仔细思考了一下,假设叫T'次时最优,那么考虑买不够T'时,一定被迫购买了保质期较长的食品,导致购买数量不如T';如果买了大于T’次,那么叫外卖的支出升高,也会导致买到的数量不够。[上面好像是一堆废话...]

      也就是说这题需要满足的是:从单峰开始往左走,每次少叫一次外卖,你的支出一定会升高;往右边走,没多叫一次外卖,你的支出一定会增高[但是真的有这个性质吗?我不这么觉得]

      下面再挂一张图:是笔者思考证明的过程。

      

      

    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    
    using namespace std;
    
    const int maxn=210;
    typedef long long ll;
    
    struct Node{
        ll s,p;
    }a[maxn],b[maxn];
    
    int n,cnt,cnt1;
    ll M,F,ans;
    bool no_use[maxn];
    
    bool cmp(const Node &A,const Node &B){
        if(A.s!=B.s) return A.s<B.s;
        return A.p<B.p;
    } 
    
    bool cmp1(const Node &A,const Node &B){
        return A.p<B.p;
    }
    
    ll get_ans(ll t){
        ll sum=M-t*F,days=0,num,res=0;
        
        for(int i=1;i<=cnt;i++){
            num=min(a[i].s-days+1,sum/t/a[i].p);//在本商品处最多能购买的天数,如果超过保质期或者没钱,那么不要 
            sum-=num*t*a[i].p,days+=num,res+=num*t;
            
            if(days<=a[i].s){//如果目前还没有超过保质期,说明没钱,那么剩下的钱全部买这个加入答案 
                num=sum/a[i].p;res+=num;return res;
            }
        }
        return res;
    }
    
    void init(){
        sort(a+1,a+n+1,cmp);
        b[++cnt1]=a[1];
        for(int i=2;i<=n;i++){
            while(a[i].s==a[i-1].s && i<n) i++;
            b[++cnt1]=a[i]; 
        }
        sort(b+1,b+cnt1+1,cmp1);
        ll tmp=b[1].s;
        for(int i=2;i<=cnt1;i++){
            if(b[i].s<=tmp) no_use[i]=true;
            else tmp=b[i].s;
        }
        for(int i=1;i<=cnt1;i++)
            if(!no_use[i])
                a[++cnt]=b[i];
    }
    
    int main(){
        freopen("3874.in","r",stdin);
        freopen("3874.out","w",stdout);
    
        scanf("%lld%lld%d",&M,&F,&n);
        for(int i=1;i<=n;i++)
            scanf("%lld%lld",&a[i].p,&a[i].s);
        init();
        
        ll l=1,r=M/(F+a[1].p),ml,mr;
        ll ansl,ansr;
        
        ans=max(get_ans(l),get_ans(r));
        while(l<=r){
            ll Len=r-l+1;
            ml=l+Len/3,mr=l+Len*2/3;
            ansl=get_ans(ml),ansr=get_ans(mr);
            if(ansl>ansr)
                ans=max(ans,ansr),r=mr-1;
            else
                ans=max(ans,ansl),l=ml+1;
        }
        
        printf("%lld",ans);
        return 0;
    }
    View Code

    [我还是莫名其妙的A了...]

    还有一个用黄金比例分割+一些特判技巧(玄学)A的?

    #include<cstdio>
    #include<cmath>
    #include<cstring>
    #include<algorithm>
    
    using namespace std;
    
    const int maxn=210;
    const double gold=(3.0-sqrt(5))/2.0;
    typedef long long ll;
    
    struct Node{
        ll s,p;
    }a[maxn],b[maxn];
    
    int n,cnt,cnt1;
    ll M,F,ans;
    bool no_use[maxn];
    
    bool cmp(const Node &A,const Node &B){
        if(A.s!=B.s) return A.s<B.s;
        return A.p<B.p;
    } 
    
    bool cmp1(const Node &A,const Node &B){
        return A.p<B.p;
    }
    
    ll get_ans(ll t){
        if(t==0) return 0;
        ll sum=M-t*F,days=0,num,res=0;
        
        for(int i=1;i<=cnt;i++){
            num=min(a[i].s-days+1,sum/t/a[i].p);//在本商品处最多能购买的天数,如果超过保质期或者没钱,那么不要 
            sum-=num*t*a[i].p,days+=num,res+=num*t;
            
            if(days<=a[i].s){//如果目前还没有超过保质期,说明没钱,那么剩下的钱全部买这个加入答案 
                num=sum/a[i].p;res+=num;return res;
            }
        }
        return res;
    }
    
    void init(){
        sort(a+1,a+n+1,cmp);
        b[++cnt1]=a[1];
        for(int i=2;i<=n;i++){
            while(a[i].s==a[i-1].s && i<n) i++;
            b[++cnt1]=a[i]; 
        }
        sort(b+1,b+cnt1+1,cmp1);
        ll tmp=b[1].s;
        for(int i=2;i<=cnt1;i++){
            if(b[i].s<=tmp) no_use[i]=true;
            else tmp=b[i].s;
        }
        for(int i=1;i<=cnt1;i++)
            if(!no_use[i])
                a[++cnt]=b[i];
    }
    
    int main(){
        scanf("%lld%lld%d",&M,&F,&n);
        for(int i=1;i<=n;i++)
            scanf("%lld%lld",&a[i].p,&a[i].s);
        init();
        
        ll l=1,r=M/(F+a[1].p),ml=1+r*gold,mr=r-r*gold;
        ll ansl=get_ans(ml),ansr=get_ans(mr);
        
        ans=max(get_ans(l),get_ans(r));
        //printf("(%lld,%lld,%lld,%lld)
    ",l,ml,mr,r);
        while(l<=r){
            if(ansl>ansr){
                ans=max(ans,ansr),r=mr-1;
                ll Len=r-l+1;
                mr=ml;ansr=ansl;
                ml=l+Len*gold;
                if(ml==mr && ml>l) ml--;
                ansl=get_ans(ml);
            }
            else{
                ans=max(ans,ansl),l=ml+1;
                ll Len=r-l+1;
                ml=mr;ansl=ansr;
                mr=r-Len*gold;
                if(mr==ml && mr<r) mr++;
                ansr=get_ans(mr);
            }
            //printf("(%lld,%lld,%lld,%lld)
    ",l,ml,mr,r);
        }
        ans=max(ans,ansl);ans=max(ans,ansr);
        printf("%lld",ans);
        return 0;
    }
    View Code

    感觉学的有点不踏实,不过至少还是学会了三分法怎么打。感谢一直帮忙的ZZD同学。Orz ZZD神犇。

    最后重申一遍:“质疑大法好!多多质疑算法的正确性!多多总结算法的适用性!”

      [第二阶段的证明思考-----2.25]

        开场首先 "Orz-TB-srO"  跪见智商帝。

        TB无意间听说我在讨论三分法,唔,就进行了愉快的思考,大喊:“这不是很容易吗?”,我当时十分愤懑= =,我思考了很久都没有想出来...居然被TB秒了。

        好吧,有的时候就是要服大神,人家毕竟思考得不一样,你看,她就看错题目了。

        她以为是宅M天需要的最少花费,难怪我和她争论了一会也不知道她到底在干什么...

        不过我想了想,感觉和原题差不多啊,如果证明出了这个是单峰函数,原题应该也是单峰函数啊。

        [然后TB埋头苦干,终于将她的思维翻译成了人类智慧层面的语言,下面是我的转述]

        首先设一个函数ans(),ans(x)表示叫x次外卖,使得其能宅M天的最小花费。

        目标:证明ans(x)是一个单峰函数。[峰值为最小值]

        现在我们再设一个函数F(),F(i)表示叫i次外卖,除掉外卖费的其它花费的最小值。

        即:F(i)=ans(i)-i*cost

        这时我们相减一下F(i)和F(i-1)

        F(i)-F(i-1)=ans(i)-ans(i-1)+cost

        如果我们希望ans()是一个下凸函数,因为下凸函数求导应该得到的斜率先是负数再到正数,而且不希望有多个凹进去的地方?

        那么ans()的导数应该是递增的才对。ans(i)-ans(i-1)正好就相当于这个导数。那么证明了F(i)-F(i-1)递增我们就证完了!

        我们考虑从F(i-1)到F(i)再到F(i+1)的过程:

        

        好了,我们现在已经得到了需要的结论,F(i)-F(i-1)是递增的。

        那么我们可以再倒着推回去一遍。

        ∵F(i)-F(i-1)递增,ans(i)-ans(i-1)=F(i)-F(i-1)+cost,cost为常数

        ∴ans(i)-ans(i-1)递增。即ans()的导函数是递增的。

        又∵ans()初始导函数为负数[一开始的时候,增加外卖次数当然是能够使最小花费变小],且结束时导函数为正数[增加外卖次数能使最小花费数增大]

        ∴ans()的导函数是先负后正的一个过程,而且ans()的导函数递增。

        ∴ans()是一个单峰函数,并存在极小值。

        然后证完啦!

        让我们再次 "Orz-TB-srO"  跪见智商帝。

        

      鸣谢Cyan提出了这么神奇的方法,鸣谢Cyan有心看蒟蒻刷的题。

      感谢笔者没有放弃思考与证明。OI有趣吖...

  • 相关阅读:
    lombok-@Accessors注解
    spring boot 当参数传入开头多个0时,报错:JSON parse error: Invalid numeric value: Leading zeroes not allowed
    linux查看历史操作记录并且显示执行时间
    IDEA中mybatis插件自动生成手写sql的xml文件
    CPU核数和load average的关系
    Jenkins--Credentials添加证书从git上拉代码
    解决输入git branch 进入编辑状态,mac下出现END,无法返回
    Git log和git reflog
    SpringCloud入门之常用的配置文件 application.yml和 bootstrap.yml区别
    springboot定时任务
  • 原文地址:https://www.cnblogs.com/Robert-Yuan/p/5215128.html
Copyright © 2011-2022 走看看