BZOJ1071 [SCOI2007]压缩 指针

欢迎访问~原文出处——博客园-zhouzhendong

去博客园看该题解


题目传送门 - BZOJ1071


题意概括

  有两个序列a[1..n], b[1..n],其编号为1..n,设为s序列。现在我们要求出最长的满足条件的s的子序列s‘,设va=min(a[s[i]]), vb=min(b[s[i]]), 满足对于所有的j=s‘[i], A*(a[j]-va)+B*(b[j]-vb)<=C。


题解

  设v[i]=A*a[i]+B*b[i];
  那么,要求满足v[s‘[i]]-A*va-B*vb<=C,
  移项得:v[s[i]]<=A*va+B*vb+C
  于是我们可以按照两种顺序排序,一个是v,一个是b。
  那么如果确定鼓励va,则:
  A*a[i]+B*b[i]<=A*va+B*vb+C
  而A*a[i]>=A*va
  所以B*b[i]<=B*vb+C
  移项B*(b[i]-vb)<=C,
  b[i]<=vb+C/B,
  都是整数,所以可以直接取整。
  所以,对于所有的b[i],有vb<=b[i]<=vb+C/B。
  然后双指针,一个扫按照b排序的,一个扫按照s排序的。


代码

#include <cstring>
#include <algorithm>
#include <cstdio>
#include <cstdlib>
#include <cmath>
using namespace std;
typedef long long LL;
const int N=5000+5;
struct Player{
	LL a,b,v;
}v[N],a[N];
int n;
LL A,B,C;
bool cmp_v(Player a,Player b){
	return a.v<b.v;
}
bool cmp_a(Player a,Player b){
	return a.a<b.a;
}
int main(){
	scanf("%d%lld%lld%lld",&n,&A,&B,&C);
	for (int i=1;i<=n;i++){
		scanf("%lld%lld",&v[i].a,&v[i].b);
		v[i].v=A*v[i].a+B*v[i].b;
		a[i]=v[i];
	}
	sort(v+1,v+n+1,cmp_v);
	sort(a+1,a+n+1,cmp_a);
	int ans=0;
	for (int i=1;i<=n;i++){
		int L=0,R=0,cnt=0;
		LL xL=a[i].b,xR=xL+C/B;
		LL va,vb=xL;
		for (int j=1;j<=n;j++){
			va=a[j].a;
			while (R<n&&v[R+1].v<=A*va+B*vb+C){
				R++;
				if (xL<=v[R].b&&v[R].b<=xR)
					cnt++;
			}
			while (L<n&&a[L+1].a<va){
				L++;
				if (xL<=a[L].b&&a[L].b<=xR)
					cnt--;
			}
			ans=max(ans,cnt);
		}
	}
	printf("%d",ans);
	return 0;
}

  

时间: 2024-12-28 21:19:08

BZOJ1071 [SCOI2007]压缩 指针的相关文章

bzoj 1068: [SCOI2007]压缩 DP

1068: [SCOI2007]压缩 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 496  Solved: 315[Submit][Status] Description 给一个由小写字母组成的字符串,我们可以用一种简单的方法来压缩其中的重复信息.压缩后的字符串除了小写字母外还可以(但不必)包含大写字母R与M,其中M标记重复串的开始,R重复从上一个M(如果当前位置左边没有M,则从串的开始算起)开始的解压结果(称为缓冲串). bcdcdcdcd可以

bzoj1068: [SCOI2007]压缩

区间dp. bool t代表区间内是否能含M. 如果不能含M的话有 res=min{f[l][i][0]+r-i}.(i<r) (串长的最小值等于前面串压缩后的最小值和不压缩后面串的长度). 如果字符串长度为偶数,且前半串等于后半串,还有 f[l][r][t]=min(f[l][(l+r)>>1][t]+1) (后半串用1个R替代). 如果t=1时,除上面俩个还有res=min{f[l][i][1]+1+f[i+1][r][1]}. 状态和3种状态转移方程比较难想.很大程度是因为对区间d

【BZOJ 1068】[SCOI2007]压缩

Description 给 一个由小写字母组成的字符串,我们可以用一种简单的方法来压缩其中的重复信息.压缩后的字符串除了小写字母外还可以(但不必)包含大写字母R与M,其中M 标记重复串的开始,R重复从上一个M(如果当前位置左边没有M,则从串的开始算起)开始的解压结果(称为缓冲串). bcdcdcdcd可以压缩为bMcdRR,下面是解压缩的过程: 另一个例子是abcabcdabcabcdxyxyz可以被压缩为abcRdRMxyRz. Input 输入仅一行,包含待压缩字符串,仅包含小写字母,长度为

BZOJ1071 [SCOI2007]组队

说好的一天一题解来啦~(其实是给马上要来的NOIP模拟赛加点RP>.<) Vfk大神说是n^2乱搞,于是蒟蒻就开始乱搞,结果发现怎么搞都是O(n ^ 3)的... 后来请教了snake大神,他说先给sum = A * h + B * s排序,然后枚举h和s的最小值.(h -- Height, s -- Speed) 然后就是两个指针乱搞: 先枚举s最小值,然后一边枚举v的最小值一边查询符合条件的人数,因为这一定是相邻的一串. 1 /*******************************

[BZOJ 1068] [SCOI2007] 压缩 【区间 DP 】

题目链接:BZOJ - 1068 题目分析 这种区间 DP 之前就做过类似的,也是字符串压缩问题,不过这道题稍微复杂一些. 需要注意如果某一段是 S1S1 重复,那么可以变成 M + Solve(S1) + R ,不过这个 Solve(S1) 中不能在中间有 M ,否则后面的 R 向前找到的 M 就不再是开头的 M 了. 代码 #include <iostream> #include <cstdio> #include <cstring> #include <al

【BZOJ】1068: [SCOI2007]压缩(dp)

http://www.lydsy.com/JudgeOnline/problem.php?id=1068 发现如果只设一维的话无法转移 那么我们开第二维,发现对于前i个来说,如果确定了M在哪里,第i个是用R还是不用就能确定了(如果用R那么在中间一定变成了缓冲串) 那么可以转移了 设d[i,j]表示前i个串,最近的一个M在i的前边一个格子,的最短长度,有 d[1,1]=1 d[i,i]=min{d[i-1,j]}+2 //即用一次M又补上i,所以+2 d[i,j]=d[pos,j]+1,其中pos

Luogu 2470 [SCOI2007]压缩

和Luogu 4302 [SCOI2003]字符串折叠 差不多的想法,区间dp 为了计算方便,我们可以假设区间[l, r]的前面放了一个M,设$f_{i, j, 0/1}$表示区间$[i, j]$中是否存在M 因为这题只能是二的幂次倍压缩,所以转移的时候枚举中点chk是否合法,如果合法那么 $f_{i, j, 0} = f_{i, (i + j) / 2 - 1, 0} + 1$ 除了区间压缩,还可以通过加法构成最优答案 1.当中间加入了M,枚举M加入的位置 $f_{i, j, 1} = min

bzoj 1068: [SCOI2007]压缩【区间dp】

神区间dp 设f[l][r][0]为在l到r中压缩的第一个字符为M,并且区间内只有这一个M,f[l][r][0]为在l到r中压缩的第一个字符为M,并且区间内有两个及以上的M 然后显然的转移是f[i][j][1]=min(f[i][k][0],f[i][k][1])+min(f[k+1][j][0],f[k+1][j][1])+1,f[i][j][0]=f[i][j][0],f[i][k][0]+j-k 然后考虑合并串,也就是当(l,mid),(mid+1,r)的串相等的时候,转移f[i][j][

B1068 [SCOI2007]压缩 区间dp

这个题我状态想对了,但是转移错了...dp的代码难度都不大,但是思考含量太高了..不会啊,我太菜了. 其实这个题就是一个正常的区间dp,中间多了一个特判的转移就行了. 题干: Description 给一个由小写字母组成的字符串,我们可以用一种简单的方法来压缩其中的重复信息.压缩后的字符串除了小 写字母外还可以(但不必)包含大写字母R与M,其中M标记重复串的开始,R重复从上一个M(如果当前位置左边没 有M,则从串的开始算起)开始的解压结果(称为缓冲串). bcdcdcdcd可以压缩为bMcdRR