bzoj 1071: [SCOI2007]组队 (单调性乱搞)

题目描述

传送门

题目大意:假如一支球队里速度最慢的球员速度为minV,身高最矮的球员高度为minH,那么这支球队的所有队员都应该满足: A(heightminH)+B(speedminV)<=C 其中A和B,C为给定的经验值。 问在N名选秀球员中,最多能有多少名符合条件的候选球员。

题解

这道题刚开始想写CDQ分治,但是minH,minV不一定出自同一个球员,所以貌似不能这么做。
这道题 O(n3) 的做法应该很好想,直接枚举minH,minV,查询符合条件的人数即可。
那么如果我们能快速的查询符合条件的人数就能将时间复杂度降到 O(n2)
A(heightminH)+B(speedminV)<=C
化简后得到 Aheight+Bspeed<=C+AminH+BminV
按照 Aheight+Bspeed 排序,先枚举minH,再从小到大枚举minV,因为 C+AminH+BminV 是单调递增的,那么能够选择地区间一定是单调移动的。
但是随着minV的递增,很有可能使原本满足 speed>=minV 的变得不满足,所以我们要提前处理掉这些情况,因为是minV从小到大枚举的,所以先枚举到的点有可能对于后面的答案有影响,那么如果此时 Aheight+BminV 是小于等于 C+AminH+BminV 的,那么在后面的计算中他一定会被加入答案,但这是不合法的所以我们需要在这个时候舍弃。

代码

#include
#include
#include
#include
#include
#define N 5006
#define LL long long 
using namespace std;
struct data{
    LL x,y,c;
}a[N],b[N];
int n; LL A,B,C;
int cmp(data a,data b){
    return a.cint cmp1(data a,data b){
    return a.yy;
}
int main()
{
    freopen("a.in","r",stdin);
    scanf("%d%lld%lld%lld",&n,&A,&B,&C);
    for (int i=1;i<=n;i++) {
        scanf("%lld%lld",&a[i].x,&a[i].y);
        a[i].c=A*a[i].x+B*a[i].y;
        b[i]=a[i];
    }
    sort(a+1,a+n+1,cmp);
    sort(b+1,b+n+1,cmp1);
    int ans=0;
    for (int i=1;i<=n;i++) {
        int top=1; int cnt=0;
        for (int j=1;j<=n;j++)
         if (b[j].x>=a[i].x&&a[i].y>=b[j].y) {
            LL mn=a[i].x*A+b[j].y*B+C;
            while (top<=n&&a[top].c<=mn) {
                if (a[top].x>=a[i].x&&a[top].y>=b[j].y) cnt++;
                top++;
             }
            ans=max(ans,cnt);
            if(b[j].c<=mn) cnt--;
         }
    }
    printf("%d\n",ans);
}

你可能感兴趣的:(乱搞)