bzoj4836 [Lydsy2017年4月月赛]二元运算

Description

定义二元运算 opt 满足

现在给定一个长为 n 的数列 a 和一个长为 m 的数列 b ,接下来有 q 次询问。每次询问给定一个数字 c 
你需要求出有多少对 (i, j) 使得 a_i  opt b_j=c 。
 
 
 

Input

第一行是一个整数 T (1≤T≤10) ,表示测试数据的组数。
对于每组测试数据:
第一行是三个整数 n,m,q (1≤n,m,q≤50000) 。
第二行是 n 个整数,表示 a_1,a_2,?,a_n (0≤a_1,a_2,?,a_n≤50000) 。
第三行是 m 个整数,表示 b_1,b_2,?,b_m (0≤b_1,b_2,?,b_m≤50000) 。
第四行是 q 个整数,第 i 个整数 c_i (0≤c_i≤100000) 表示第 i 次查询的数。
 
 

Output

对于每次查询,输出一行,包含一个整数,表示满足条件的 (i, j) 对的个数。

 
 

Sample Input

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

Sample Output

1
0
1
0
0
1
0
1
0
1

 

正解:分治$FFT$。

比较简单的一个题。

构造$a,b$的生成函数,$a[i]$表示$a$序列中$i$出现的次数。

考虑分治,我们每次把区间分成两半时,跑两次$FFT$。

对于第一种情况,我们直接把$a$的$[l,mid]$部分和$b$的$[mid+1,r]$卷积即可。

对于第二种情况,我们考虑把$b$序列的$[l,mid]$部分翻转,那么就是$a$的$[mid+1,r]$部分和翻转以后的$b$的$[l,mid]$部分卷积得到的多项式。

然后我们递归两个子区间就行了。

还有两个序列中元素相等的情况,直接特判就行了。

 

 1 #include <bits/stdc++.h>
 2 #define il inline
 3 #define RG register
 4 #define ll long long
 5 #define N (500010)
 6 
 7 using namespace std;
 8 
 9 struct data{ double x,y; }A[N],B[N];
10 
11 const double pi=acos(-1.0);
12 int a[N],b[N],rev[N],n,m,q,Mx,len1,len2;
13 ll ans[N];
14 
15 il int gi(){
16   RG int x=0,q=1; RG char ch=getchar();
17   while ((ch<'0' || ch>'9') && ch!='-') ch=getchar();
18   if (ch=='-') q=-1,ch=getchar();
19   while (ch>='0' && ch<='9') x=x*10+ch-48,ch=getchar();
20   return q*x;
21 }
22 
23 il data operator + (const data &a,const data &b){
24   return (data){a.x+b.x,a.y+b.y};
25 }
26 
27 il data operator - (const data &a,const data &b){
28   return (data){a.x-b.x,a.y-b.y};
29 }
30 
31 il data operator * (const data &a,const data &b){
32   return (data){a.x*b.x-a.y*b.y,a.x*b.y+a.y*b.x};
33 }
34 
35 il void fft(data *a,RG int n,RG int f){
36   for (RG int i=0;i<n;++i) if (i<rev[i]) swap(a[i],a[rev[i]]);
37   for (RG int i=1;i<n;i<<=1){
38     RG data wn=(data){cos(pi/i),sin(pi/i)},x,y;
39     for (RG int j=0;j<n;j+=i<<1){
40       RG data w=(data){1,0};
41       for (RG int k=0;k<i;++k,w=w*wn){
42     x=a[j+k],y=w*a[j+k+i];
43     a[j+k]=x+y,a[j+k+i]=x-y;
44       }
45     }
46   }
47   if (f==1) return; reverse(a+1,a+n);
48   for (RG int i=0;i<n;++i) a[i].x/=n; return;
49 }
50 
51 il void solve(RG int l,RG int r){
52   if (l==r) return; RG int mid=(l+r)>>1,len,lg=0;
53   for (len=1;len<=r-l+1;len<<=1) ++lg;
54   for (RG int i=0;i<len;++i) rev[i]=rev[i>>1]>>1|((i&1)<<(lg-1));
55   for (RG int i=0;i<len;++i) A[i]=B[i]=(data){0,0};  
56   for (RG int i=l;i<=mid;++i) A[i-l].x=a[i]; fft(A,len,1);
57   for (RG int i=mid+1;i<=r;++i) B[i-mid-1].x=b[i]; fft(B,len,1);
58   for (RG int i=0;i<len;++i) A[i]=A[i]*B[i]; fft(A,len,-1);
59   for (RG int i=0;i<len;++i) ans[i+mid+1+l]+=(ll)(A[i].x+0.5);
60   for (RG int i=0;i<len;++i) A[i]=B[i]=(data){0,0};
61   for (RG int i=mid+1;i<=r;++i) A[i-mid-1].x=a[i]; fft(A,len,1);
62   for (RG int i=l;i<=mid;++i) B[mid-i].x=b[i]; fft(B,len,1);
63   for (RG int i=0;i<len;++i) A[i]=A[i]*B[i]; fft(A,len,-1);
64   for (RG int i=0;i<len;++i) ans[i+1]+=(ll)(A[i].x+0.5);
65   solve(l,mid),solve(mid+1,r); return;
66 }
67 
68 il void work(){
69   n=gi(),m=gi(),q=gi(),memset(ans,0,sizeof(ans));
70   memset(a,0,sizeof(a)),memset(b,0,sizeof(b)),len1=len2=0;
71   for (RG int i=1,x;i<=n;++i) ++a[x=gi()],len1=max(len1,x);
72   for (RG int i=1,x;i<=m;++i) ++b[x=gi()],len2=max(len2,x);
73   for (RG int i=1;i<=len1 && i<=len2;++i) ans[0]+=1LL*a[i]*b[i];
74   Mx=max(len1,len2),solve(0,Mx);
75   for (RG int i=1;i<=q;++i) printf("%lld\n",ans[gi()]); return;
76 }
77 
78 int main(){
79 #ifndef ONLINE_JUDGE
80   freopen("calc.in","r",stdin);
81   freopen("calc.out","w",stdout);
82 #endif
83   RG int T=gi();
84   while (T--) work();
85   return 0;
86 }

 

posted @ 2017-08-23 16:26  wfj_2048  阅读(163)  评论(0编辑  收藏  举报