【题目链接】 http://acm.hdu.edu.cn/showproblem.php?pid=6085
【题目大意】
给出一个数组a一个数组b,以及询问数组c,
问对于每个c有多少对a%b=c,答案对2取模
【题解】
考虑对2取模我们发现答案等价于数字的xor,01状态可以用bitset保存,
在bitset上存a的权值数组,那么每次只要将b*i~b*(i+1)-1的数值xor到答案数组的0~b-1上去即可,
鉴于bitset没有截取区间的功能,我们手写压位,
考虑压32位之后不是32倍数的部分截取起来非常尴尬,
因此我们保存其偏移量为0~31的数组,这样取区间就比较方便了。
【代码】
#include <cstdio> #include <algorithm> #include <cstring> using namespace std; const int N=10010; struct Bitset{ unsigned int u[N]; void reset(){memset(u,0,sizeof(u));} void set(int x){u[x>>5]|=1<<(x&31);} void flip(int x){u[x>>5]^=1<<(x&31);} bool test(int x){return u[x>>5]&(1<<(x&31));} void reset(int x){if(test(x))flip(x);} }a[32],ans; void Solve(int l,int r){ while((r-l)&31){r--;if(a[0].test(r))ans.flip(r-l);} int m=0; while(l&31)l++,r++,m++; l>>=5; r>>=5; for(int i=l;i<r;i++)ans.u[i-l]^=a[m].u[i]; } int T,n,m,q,x,mx; int main(){ scanf("%d",&T); while(T--){ scanf("%d%d%d",&n,&m,&q); for(int i=0;i<32;i++)a[i].reset(); ans.reset(); mx=0; for(int i=1;i<=n;i++){ scanf("%d",&x); mx=max(x,mx); for(int j=0;j<32;j++)a[j].set(x+j); } for(int i=1;i<=m;i++){ scanf("%d",&x); for(int j=0;j<=mx;j+=x)Solve(j,min(mx+1,j+x)); } while(q--){ scanf("%d",&x); if(ans.test(x))puts("1"); else puts("0"); } }return 0; }