【bzoj4810】[Ynoi2017]由乃的玉米田 莫队算法+STL-bitset

题目描述

由乃在自己的农田边散步,她突然发现田里的一排玉米非常的不美。这排玉米一共有N株,它们的高度参差不齐。
由乃认为玉米田不美,所以她决定出个数据结构题
这个题是这样的:
给你一个序列a,长度为n,有m次操作,每次询问一个区间是否可以选出两个数它们的差为x,或者询问一个区间是
否可以选出两个数它们的和为x,或者询问一个区间是否可以选出两个数它们的乘积为x ,这三个操作分别为操作1
,2,3选出的这两个数可以是同一个位置的数

输入

第一行两个数n,m
后面一行n个数表示ai
后面m行每行四个数opt l r x
opt表示这个是第几种操作,l,r表示操作的区间,x表示这次操作的x
定义c为每次的x和ai中的最大值,ai >= 0,每次的x>=2n,m,c <= 100000

输出

对于每个询问,如果可以,输出yuno,否则输出yumi

样例输入

5 5
1 1 2 3 4
2 1 1 2
1 1 2 2
3 1 1 1
3 5 5 16
1 2 3 4

样例输出

yuno
yumi
yuno
yuno
yumi


题解

莫队算法+STL-bitset

先将询问离线排序,对于每个询问,把它区间内包含的所有数放在一个桶里。

如果是3询问,直接暴力O(√n)枚举约数,即可。

对于询问1,一个naive的做法是枚举每个数,判断是否有比它小x的数。

我们可以使用bitset压位,优化这个过程,将bitset数组左移或右移x位并与原数组取与,判断是否存在。

对于询问2,将a+b=x转化为(100000-a)-b=100000-x后按照同样的方法处理。

#include <cstdio>
#include <cmath>
#include <bitset>
#include <algorithm>
#define N 100010
using namespace std;
bitset<N> a , b;
struct data
{
	int opt , l , r , x , bl , id;
}q[N];
int v[N] , cnt[N] , ans[N];
bool cmp(data a , data b)
{
	return a.bl == b.bl ? a.r < b.r : a.bl < b.bl;
}
bool judge(int x)
{
	int i;
	for(i = 1 ; i * i <= x ; i ++ )
		if(x % i == 0 && cnt[i] && cnt[x / i])
			return 1;
	return 0;
}
int main()
{
	int n , m , si , i , lp = 1 , rp = 0;
	scanf("%d%d" , &n , &m) , si = (int)sqrt(n);
	for(i = 1 ; i <= n ; i ++ ) scanf("%d" , &v[i]);
	for(i = 1 ; i <= m ; i ++ ) scanf("%d%d%d%d" , &q[i].opt , &q[i].l , &q[i].r , &q[i].x) , q[i].bl = (q[i].l - 1) / si , q[i].id = i;
	sort(q + 1 , q + m + 1 , cmp);
	for(i = 1 ; i <= m ; i ++ )
	{
		while(rp < q[i].r) cnt[v[++rp]] ++ , a[v[rp]] = b[100000 - v[rp]] = 1;
		while(lp > q[i].l) cnt[v[--lp]] ++ , a[v[lp]] = b[100000 - v[lp]] = 1;
		while(rp > q[i].r) cnt[v[rp]] -- , a[v[rp]] = b[100000 - v[rp]] = cnt[v[rp]] , rp -- ;
		while(lp < q[i].l) cnt[v[lp]] -- , a[v[lp]] = b[100000 - v[lp]] = cnt[v[lp]] , lp ++ ;
		if(q[i].opt == 1) ans[q[i].id] = (a & (a >> q[i].x)).any();
		else if(q[i].opt == 2) ans[q[i].id] = (a & (b >> (100000 - q[i].x))).any();
		else ans[q[i].id] = judge(q[i].x);
	}
	for(i = 1 ; i <= m ; i ++ ) printf("%s\n" , ans[i] ? "yuno" : "yumi");
	return 0;
}

 

posted @ 2017-06-13 13:29  GXZlegend  阅读(450)  评论(0编辑  收藏  举报