题目描述
传送门
题目大意:假如一支球队里速度最慢的球员速度为minV,身高最矮的球员高度为minH,那么这支球队的所有队员都应该满足: A∗(height–minH)+B∗(speed–minV)<=C 其中A和B,C为给定的经验值。 问在N名选秀球员中,最多能有多少名符合条件的候选球员。
题解
这道题刚开始想写CDQ分治,但是minH,minV不一定出自同一个球员,所以貌似不能这么做。
这道题
O(n3)
的做法应该很好想,直接枚举minH,minV,查询符合条件的人数即可。
那么如果我们能快速的查询符合条件的人数就能将时间复杂度降到
O(n2)
A∗(height–minH)+B∗(speed–minV)<=C
化简后得到
A∗height+B∗speed<=C+A∗minH+B∗minV
按照
A∗height+B∗speed
排序,先枚举minH,再从小到大枚举minV,因为
C+A∗minH+B∗minV
是单调递增的,那么能够选择地区间一定是单调移动的。
但是随着minV的递增,很有可能使原本满足
speed>=minV
的变得不满足,所以我们要提前处理掉这些情况,因为是minV从小到大枚举的,所以先枚举到的点有可能对于后面的答案有影响,那么如果此时
A∗height+B∗minV
是小于等于
C+A∗minH+B∗minV
的,那么在后面的计算中他一定会被加入答案,但这是不合法的所以我们需要在这个时候舍弃。
代码
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#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.c<b.c;
}
int cmp1(data a,data b){
return a.y<b.y;
}
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);
}