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]$部分卷积得到的多项式。

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

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

 #include <bits/stdc++.h>
#define il inline
#define RG register
#define ll long long
#define N (500010) using namespace std; struct data{ double x,y; }A[N],B[N]; const double pi=acos(-1.0);
int a[N],b[N],rev[N],n,m,q,Mx,len1,len2;
ll ans[N]; il int gi(){
RG int x=,q=; RG char ch=getchar();
while ((ch<'' || ch>'') && ch!='-') ch=getchar();
if (ch=='-') q=-,ch=getchar();
while (ch>='' && ch<='') x=x*+ch-,ch=getchar();
return q*x;
} il data operator + (const data &a,const data &b){
return (data){a.x+b.x,a.y+b.y};
} il data operator - (const data &a,const data &b){
return (data){a.x-b.x,a.y-b.y};
} il data operator * (const data &a,const data &b){
return (data){a.x*b.x-a.y*b.y,a.x*b.y+a.y*b.x};
} il void fft(data *a,RG int n,RG int f){
for (RG int i=;i<n;++i) if (i<rev[i]) swap(a[i],a[rev[i]]);
for (RG int i=;i<n;i<<=){
RG data wn=(data){cos(pi/i),sin(pi/i)},x,y;
for (RG int j=;j<n;j+=i<<){
RG data w=(data){,};
for (RG int k=;k<i;++k,w=w*wn){
x=a[j+k],y=w*a[j+k+i];
a[j+k]=x+y,a[j+k+i]=x-y;
}
}
}
if (f==) return; reverse(a+,a+n);
for (RG int i=;i<n;++i) a[i].x/=n; return;
} il void solve(RG int l,RG int r){
if (l==r) return; RG int mid=(l+r)>>,len,lg=;
for (len=;len<=r-l+;len<<=) ++lg;
for (RG int i=;i<len;++i) rev[i]=rev[i>>]>>|((i&)<<(lg-));
for (RG int i=;i<len;++i) A[i]=B[i]=(data){,};
for (RG int i=l;i<=mid;++i) A[i-l].x=a[i]; fft(A,len,);
for (RG int i=mid+;i<=r;++i) B[i-mid-].x=b[i]; fft(B,len,);
for (RG int i=;i<len;++i) A[i]=A[i]*B[i]; fft(A,len,-);
for (RG int i=;i<len;++i) ans[i+mid++l]+=(ll)(A[i].x+0.5);
for (RG int i=;i<len;++i) A[i]=B[i]=(data){,};
for (RG int i=mid+;i<=r;++i) A[i-mid-].x=a[i]; fft(A,len,);
for (RG int i=l;i<=mid;++i) B[mid-i].x=b[i]; fft(B,len,);
for (RG int i=;i<len;++i) A[i]=A[i]*B[i]; fft(A,len,-);
for (RG int i=;i<len;++i) ans[i+]+=(ll)(A[i].x+0.5);
solve(l,mid),solve(mid+,r); return;
} il void work(){
n=gi(),m=gi(),q=gi(),memset(ans,,sizeof(ans));
memset(a,,sizeof(a)),memset(b,,sizeof(b)),len1=len2=;
for (RG int i=,x;i<=n;++i) ++a[x=gi()],len1=max(len1,x);
for (RG int i=,x;i<=m;++i) ++b[x=gi()],len2=max(len2,x);
for (RG int i=;i<=len1 && i<=len2;++i) ans[]+=1LL*a[i]*b[i];
Mx=max(len1,len2),solve(,Mx);
for (RG int i=;i<=q;++i) printf("%lld\n",ans[gi()]); return;
} int main(){
#ifndef ONLINE_JUDGE
freopen("calc.in","r",stdin);
freopen("calc.out","w",stdout);
#endif
RG int T=gi();
while (T--) work();
return ;
}

bzoj4836 [Lydsy2017年4月月赛]二元运算的更多相关文章

  1. bzoj 4836: [Lydsy2017年4月月赛]二元运算 -- 分治+FFT

    4836: [Lydsy2017年4月月赛]二元运算 Time Limit: 8 Sec  Memory Limit: 128 MB Description 定义二元运算 opt 满足   现在给定一 ...

  2. 【bzoj4836】[Lydsy2017年4月月赛]二元运算 分治+FFT

    题目描述 定义二元运算 opt 满足   现在给定一个长为 n 的数列 a 和一个长为 m 的数列 b ,接下来有 q 次询问.每次询问给定一个数字 c  你需要求出有多少对 (i, j) 使得 a_ ...

  3. [补档][Lydsy2017年4月月赛]抵制克苏恩

    [Lydsy2017年4月月赛]抵制克苏恩 题目 小Q同学现在沉迷炉石传说不能自拔.他发现一张名为克苏恩的牌很不公平. 如果你不玩炉石传说,不必担心,小Q同学会告诉你所有相关的细节.炉石传说是这样的一 ...

  4. 【BZOJ 4832 】 4832: [Lydsy2017年4月月赛]抵制克苏恩 (期望DP)

    4832: [Lydsy2017年4月月赛]抵制克苏恩 Time Limit: 1 Sec  Memory Limit: 128 MBSubmit: 275  Solved: 87 Descripti ...

  5. 【BZOJ4832】[Lydsy2017年4月月赛]抵制克苏恩 概率与期望

    [BZOJ4832][Lydsy2017年4月月赛]抵制克苏恩 Description 小Q同学现在沉迷炉石传说不能自拔.他发现一张名为克苏恩的牌很不公平.如果你不玩炉石传说,不必担心,小Q同学会告诉 ...

  6. 【BZOJ4883】[Lydsy2017年5月月赛]棋盘上的守卫 KM算法

    [BZOJ4883][Lydsy2017年5月月赛]棋盘上的守卫 Description 在一个n*m的棋盘上要放置若干个守卫.对于n行来说,每行必须恰好放置一个横向守卫:同理对于m列来说,每列 必须 ...

  7. BZOJ 4881: [Lydsy2017年5月月赛]线段游戏

    4881: [Lydsy2017年5月月赛]线段游戏 Time Limit: 3 Sec  Memory Limit: 256 MBSubmit: 164  Solved: 81[Submit][St ...

  8. [Bzoj4832][Lydsy2017年4月月赛]抵制克苏恩 (期望dp)

    4832: [Lydsy2017年4月月赛]抵制克苏恩 Time Limit: 1 Sec  Memory Limit: 128 MBSubmit: 673  Solved: 261[Submit][ ...

  9. 抵制克苏恩[Lydsy2017年4月月赛]

    题目描述 小Q同学现在沉迷炉石传说不能自拔.他发现一张名为克苏恩的牌很不公平.如果你不玩炉石传说,不必担心,小Q同学会告诉你所有相关的细节.炉石传说是这样的一个游戏,每个玩家拥有一个 30 点血量的英 ...

随机推荐

  1. sqlserver 服务器监控

    1.表锁 查看被锁的表:select request_session_id spid,OBJECT_NAME(resource_associated_entity_id) tableName from ...

  2. 12.标准标签库(JSTL)

    1.JSTL标签库安装 JSTL的概念: JSP标准标签库(JSP Standard Tag Library)是一个实现 Web 应用程序中常见的通用功能的定制标记库集功能包括迭代和条件判断.数据管理 ...

  3. 6.过滤器(Filter)

    ---恢复内容开始--- 1.过滤器简介: 过滤器是一个用于拦截在数据源和数据目的地之间消息的一个对象. 过滤器 功能: 分析请求,将请求发送给指定的资源或自己创建一个响应返回:在请求到达服务器端前处 ...

  4. php数组·的方法-数组检索

    /* * //数组检索函数 * */ //array_keys() 获取数组中所有键名 //array_values() 获取数组中所有键名 $arr6=range('a','e'); print_r ...

  5. my.副本

    梦幻西游手游 天蓬下凡 副本 第三关:天蓬终于决定要告白了,主怪大唐,护卫狮驼岭,帮凶是龙宫,打过这关也很简单. 第四关:康太尉找来了天兵,怪仍然是有法有攻,但是打起来也是不难. ZC: 姚太尉是物理 ...

  6. Linux环境编程--waitpid与fork与execlp

    waitpid waitpid(等待子进程中断或结束) 表头文件 #include<sys/types.h> #include<sys/wait.h> 定义函数 pid_t w ...

  7. 牛客网练习赛28A

    题目链接:https://www.nowcoder.com/acm/contest/200/A 链接:https://www.nowcoder.com/acm/contest/200/A来源:牛客网 ...

  8. Unity ContextMenu 上下文菜单

    新建脚本: public class ContextTesting : MonoBehaviour { [ContextMenu("哈哈")] void DoSomething() ...

  9. UGUI Slider

    1.新建一个Cube命名为Player,在上面挂一个脚本命名为Player,脚本内容如下: using System.Collections; using System.Collections.Gen ...

  10. Java集合框架概述

    集合框架指的是容器类.Java中大量持有对象的方式有数组和容器类两种方式.数组相较于容器类的优点在于:①随机访问效率高:由于是连续的存储空间,可以计算地址直接访问 ②类型确定:数组在创建时即可确定元素 ...