聪明的质检员

题面

由于我们并不清楚要求的W的值,但是我们知道W的值不超过矿石中价值最大的,如果W大于了矿石中价值最大的,那么Y的值为0,无法达到最优解。

因此,很容易就能想到在确定W的值要用二分的方法。

在分析这道题的时候,我们很容易知道Y的值是满足单调性的,当W的值越大,Y的值越小,因为W越大,能够选的矿石就越少。

所以我们把得到的Y值作为判断条件,如果Y比S小,就说明检验值了,而W取大了。每次更改W的同时给ans取最小值。

那么Y又应该怎么求出呢?题目中n,m最大有2*10^5,

如果暴搜肯定超时,因此我们需要在枚举W的时候预处理。遍历w数组,保存满足条件的v和个数的前缀和。然后再遍历一遍要求的区间。

即sum += ( cnt[R[i]] - cnt[L[i]-1] )*( sumv[R[i]] - sumv[L[i]] )

总的时间复杂度就从O (mnlog(maxw)) 变成了O((m+n)*log(maxw))

#include<cmath>
#include<cstdio>
#include<string>
#include<cstring>
#include<iostream>
#include<algorithm>
#define maxn 2000010
#define int long long
using namespace std;
int n,m,r=0,l=214748364,mid,ans=999999999999999,s,sum,w[maxn],v[maxn],rt[maxn],lf[maxn],pv[maxn],pn[maxn];
inline bool ck(int x){
    memset(pv,0,sizeof(pv));
    memset(pn,0,sizeof(pn));
    for(int i=1;i<=n;i++)
        if(w[i]>=x){
            pv[i]=pv[i-1]+v[i];
            pn[i]=pn[i-1]+1;
        }
        else{
            pv[i]=pv[i-1];
            pn[i]=pn[i-1];
        }
    int y=0;
    for(int i=1;i<=m;i++)
        y+=(pn[rt[i]]-pn[lf[i]-1])*(pv[rt[i]]-pv[lf[i]-1]);
    sum=abs(y-s);
    if(y>s) return 1;
    return 0;
}
signed main(){
    scanf("%lld%lld%lld",&n,&m,&s);
    for(int i=1;i<=n;i++) scanf("%lld%lld",&w[i],&v[i]),r=max(r,w[i]),l=min(l,w[i]);
    for(int i=1;i<=m;i++) scanf("%lld%lld",&lf[i],&rt[i]);
    l-=1,r+=2;
    while(l<=r){
        mid=(l+r)>>1;
        if(ck(mid)) l=mid+1;
        else r=mid-1;
        ans=min(ans,sum);
    }
    ans=min(ans,sum);
    printf("%lld",ans);
    return 0;
}

  

posted @ 2019-09-15 13:54  [jackeylove]  阅读(133)  评论(0编辑  收藏  举报