CF939E:Maximize! ——题解

http://codeforces.com/problemset/problem/939/E

https://vjudge.net/problem/CodeForces-939E

给一个集合,每次两个操作:

1 a:将一个大于集合所有数的a添加进集合。

2:查询集合中所有子集的最大值-平均值的最大值。

有一个结论:最大值所在的子集一定包含集合最大值,我们用感性证明证明一下。

我们假设结论为假,那么我们已经选定了一个子集,设其最大值为x,我们在集合里还有一个数为x+1,则如果将x换成x+1,平均值只会加上一个小于1的数,而最大值却加1,显然要比原先的值大,所以x+1更优。

同理x与x+2比较可以看成x+1与x+2比较,以此类推。

所以结论必为真。

那么既然我们能固定一个最大值,我们还有一个显然成立的结论,就是子集一定为最小的前k个数和最大的数x构成。因为太显然了就不证了。

既然这样,那么显然答案是一个单峰函数,可以三分求解。

(第一次碰这个题的时候前几个步骤都想到了三分愣是没想到……我是真的蠢,看样子题刷少了)

#include<cstdio>
#include<algorithm>
using namespace std;
typedef double dl;
typedef long long ll;
const int Q=5e5+5;
ll sum[Q],maxx;
int len=0;
inline dl suan(int k){
    return 1.0*(maxx*k-sum[k])/(k+1);
}
dl sanfen(int l,int r){
    int mid1,mid2;
    while(l<=r){
        if(r-l<3){
            dl ans=suan(r);
            for(int i=l;i<r;i++)ans=max(ans,suan(i));
            return ans;
        }
        mid1=l+(r-l)/3,mid2=mid1+(r-l)/3;
        if(suan(mid1)>suan(mid2))r=mid2;
        else l=mid1;
    }
}
int main(){
    int q,op;
    scanf("%d",&q);
    while(q--){
        scanf("%d",&op);
        if(op==1){
            scanf("%lld",&maxx);
            sum[++len]=sum[len-1]+maxx;
        }
        else printf("%.10lf\n",sanfen(0,len-1));
    }
    return 0;
}

+++++++++++++++++++++++++++++++++++++++++++

 +本文作者:luyouqi233。               +

 +欢迎访问我的博客:http://www.cnblogs.com/luyouqi233/+

+++++++++++++++++++++++++++++++++++++++++++

posted @ 2018-02-21 11:58  luyouqi233  阅读(418)  评论(0编辑  收藏  举报