Description
Input
Output
Sample Input
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
Sample Output
yumi
yuno
yuno
yumi
题解:区间询问当然是用莫队来水啊~
对于差为x,直接在bitset上搞就行了;对于和为x,我们维护一个翻转的bitset,然后转化成差为x的问题来搞;对于乘积为x,我们直接开个桶,用sqrt(x)的复杂度枚举x的约数就行了。
小号交题一次提交顺利成为status倒数第一,大号提交先TLE了两次,然后加了一堆优化后顺利成为status倒数第二,试问还有谁~~~所以千万不要拿我的代码当做标程来拍。
#include <cstdio> #include <iostream> #include <cstring> #include <bitset> #include <algorithm> #include <cmath> using namespace std; const int maxn=100010; bitset<maxn*2> b1,b2; int n,m,B; int v[maxn],st[maxn*2],ans[maxn]; struct QUERY { int qa,qb,qk,qx,org; }q[maxn]; bool cmp(QUERY a,QUERY b) { return (a.qa/B==b.qa/B)?(a.qb<b.qb):(a.qa/B<b.qa/B); } void ins(int x) { if(!st[x]) b1[x]=b2[100000-x]=1; st[x]++; } void del(int x) { st[x]--; if(!st[x]) b1[x]=b2[100000-x]=0; } int rd() { int ret=0,f=1; char gc=getchar(); while(gc<'0'||gc>'9') {if(gc=='-')f=-f; gc=getchar();} while(gc>='0'&&gc<='9') ret=ret*10+gc-'0',gc=getchar(); return ret*f; } int main() { n=rd(),m=rd(); B=(int)sqrt((double)n); int i,j,l=1,r=0; for(i=1;i<=n;i++) v[i]=rd(); for(i=1;i<=m;i++) q[i].qk=rd(),q[i].qa=rd(),q[i].qb=rd(),q[i].qx=rd(),q[i].org=i; sort(q+1,q+m+1,cmp); for(i=1;i<=m;i++) { while(l>q[i].qa) ins(v[--l]); while(r<q[i].qb) ins(v[++r]); while(l<q[i].qa) del(v[l++]); while(r>q[i].qb) del(v[r--]); if(q[i].qk==1) ans[q[i].org]=((b1<<q[i].qx)&b1).any(); if(q[i].qk==2) ans[q[i].org]=((b1<<100000)&(b2<<q[i].qx)).any(); if(q[i].qk==3) for(j=1;j*j<=q[i].qx&&!ans[q[i].org];j++) if(q[i].qx%j==0&&st[j]&&st[q[i].qx/j]) ans[q[i].org]=1; } for(i=1;i<=m;i++) { if(ans[i]) printf("yuno\n"); else printf("yumi\n"); } return 0; }
| 欢迎来原网站坐坐! >原文链接<