[BZOJ4836]二元运算(分治FFT)
4836: [Lydsy1704月赛]二元运算
Time Limit: 8 Sec Memory Limit: 128 MB
Submit: 578 Solved: 202
[Submit][Status][Discuss]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 5Sample Output
1
0
1
0
0
1
0
1
0
1HINT
Source
挺有意思的一道题,应该不难想到是分治FFT,主要是CDQ分治函数内部要想得很清楚。
首先如果只有加法,就是裸卷积。
如果只有减法,那么有$c_k=\sum\limits_{i=k}^na_ib_{i-k}$,把b翻转,最后将c前移n位即可$c_{n+k}=\sum\limits_{i=0}^na_{k+i}b_{n-i}$
现在有了大小关系的限制,我们可以通过$CDQ$分治处理。
首先特判掉相等的情况,然后对于$[l,r]$这个区间,将$[l,mid]$和$[mid+1,r]$递归处理,现在问题只剩下$a[l,mid]$和$b[mid+1,r]$,$a[mid+1,r]$和$b[l,mid]$两个问题了。
显然前一个问题直接将$a$前移$l$位,$b$前移$mid+1$位,卷起来之后再后移$l+mid+1$位即可。后一个问题,将$a$前移$mid+1$位,b翻转后前移$l$位,这样卷之后的区间下标是从$0$开始的,而我们的差是从$1$开始的,所以右移$1$位。
卷的时候还是一定要注意次数界!NTT也可以。
#include<cmath>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define mem(a) memset(a,0,sizeof(a))
typedef long long ll;
using namespace std; const int N=;
const double pi=acos(-.);
struct C{ double x,y; }A[N],B[N];
int a[N],b[N],rev[N],n,T,x,m,Q,mx,len1,len2;
ll ans[N]; C operator +(C &a,C &b){ return (C){a.x+b.x,a.y+b.y}; }
C operator -(C &a,C &b){ return (C){a.x-b.x,a.y-b.y}; }
C operator *(const C &a,const C &b){ return (C){a.x*b.x-a.y*b.y,a.x*b.y+a.y*b.x}; } void DFT(C a[],int n,int f){
for (int i=; i<n; i++) if (i<rev[i]) swap(a[i],a[rev[i]]);
for (int i=; i<n; i<<=){
C wn=(C){cos(pi/i),f*sin(pi/i)};
for (int p=i<<,j=; j<n; j+=p){
C w=(C){,};
for (int k=; k<i; k++,w=w*wn){
C x=a[j+k],y=w*a[i+j+k]; a[j+k]=x+y; a[i+j+k]=x-y;
}
}
}
if (f==) return;
for (int i=; i<n; i++) a[i].x/=n;
} void CDQ(int l,int r){
if (l==r) return;
int mid=(l+r)>>,n,L=;
for (n=; n<=r-l+; n<<=) L++;
for (int i=; i<n; i++) rev[i]=(rev[i>>]>>)|((i&)<<(L-)); for (int i=; i<n; i++) A[i]=B[i]=(C){,};
for (int i=l; i<=mid; i++) A[i-l].x=a[i];
for (int i=mid+; i<=r; i++) B[i-mid-].x=b[i];
DFT(A,n,); DFT(B,n,);
for (int i=; i<n; i++) A[i]=A[i]*B[i];
DFT(A,n,-);
for (int i=; i<n; i++) ans[i+mid++l]+=(ll)(A[i].x+0.5); for (int i=; i<n; i++) A[i]=B[i]=(C){,};
for (int i=mid+; i<=r; i++) A[i-mid-].x=a[i];
for (int i=l; i<=mid; i++) B[mid-i].x=b[i];
DFT(A,n,); DFT(B,n,);
for (int i=; i<n; i++) A[i]=A[i]*B[i];
DFT(A,n,-);
for (int i=; i<n; i++) ans[i+]+=(ll)(A[i].x+0.5);
CDQ(l,mid); CDQ(mid+,r);
} int main(){
freopen("bzoj4836.in","r",stdin);
freopen("bzoj4836.out","w",stdout);
for (scanf("%d",&T); T--; ){
scanf("%d%d%d",&n,&m,&Q); mem(ans); mem(a); mem(b); len1=len2=;
for (int i=; i<=n; i++) scanf("%d",&x),a[x]++,len1=max(len1,x);
for (int i=; i<=m; i++) scanf("%d",&x),b[x]++,len2=max(len2,x);
for (int i=; i<=len1 && i<=len2; i++) ans[]+=1ll*a[i]*b[i];
CDQ(,max(len1,len2));
for (int i=; i<=Q; i++) scanf("%d",&x),printf("%lld\n",ans[x]);
}
return ;
}
[BZOJ4836]二元运算(分治FFT)的更多相关文章
- 【bzoj4836】[Lydsy2017年4月月赛]二元运算 分治+FFT
题目描述 定义二元运算 opt 满足 现在给定一个长为 n 的数列 a 和一个长为 m 的数列 b ,接下来有 q 次询问.每次询问给定一个数字 c 你需要求出有多少对 (i, j) 使得 a_ ...
- bzoj 4836 [Lydsy1704月赛]二元运算 分治FFT+生成函数
[Lydsy1704月赛]二元运算 Time Limit: 8 Sec Memory Limit: 128 MBSubmit: 577 Solved: 201[Submit][Status][Di ...
- 【bzoj4836】二元运算 分治FFT
Description 定义二元运算 opt 满足 现在给定一个长为 n 的数列 a 和一个长为 m 的数列 b ,接下来有 q 次询问.每次询问给定一个数字 c 你需要求出有多少对 (i, j) 使 ...
- bzoj 4836: [Lydsy2017年4月月赛]二元运算 -- 分治+FFT
4836: [Lydsy2017年4月月赛]二元运算 Time Limit: 8 Sec Memory Limit: 128 MB Description 定义二元运算 opt 满足 现在给定一 ...
- BZOJ 4836: [Lydsy1704月赛]二元运算 分治FFT
Code: #include<bits/stdc++.h> #define ll long long #define maxn 500000 #define setIO(s) freope ...
- BZOJ4836 [Lydsy1704月赛]二元运算 分治 多项式 FFT
原文链接http://www.cnblogs.com/zhouzhendong/p/8830036.html 题目传送门 - BZOJ4836 题意 定义二元运算$opt$满足 $$x\ opt\ y ...
- BZOJ4836: [Lydsy1704月赛]二元运算【分治FFT】【卡常(没卡过)】
Description 定义二元运算 opt 满足 现在给定一个长为 n 的数列 a 和一个长为 m 的数列 b ,接下来有 q 次询问.每次询问给定一个数字 c 你需要求出有多少对 (i, j) 使 ...
- BNUOJ 51279[组队活动 Large](cdq分治+FFT)
传送门 大意:ACM校队一共有n名队员,从1到n标号,现在n名队员要组成若干支队伍,每支队伍至多有m名队员,求一共有多少种不同的组队方案.两个组队方案被视为不同的,当且仅当存在至少一名队员在两种方案中 ...
- hdu 5730 Shell Necklace [分治fft | 多项式求逆]
hdu 5730 Shell Necklace 题意:求递推式\(f_n = \sum_{i=1}^n a_i f_{n-i}\),模313 多么优秀的模板题 可以用分治fft,也可以多项式求逆 分治 ...
随机推荐
- PHP 练习2:投票
1.建立数据库 表1:DiaoYanTiMu 表2:DiaoYanXuanXiang 2.页面 页面1:投票首页 <!DOCTYPE html PUBLIC "-//W3C//DTD ...
- ajax中datatype的json和jsonp
前言: 说到AJAX就会不可避免的面临两个问题,第一个是AJAX以何种格式来交换数据?第二个是跨域的需求如何解决?这两个问题目前都有不同的解决方案,比如数据可以用自定义字符串或者用XML来描述,跨域 ...
- Vue 双向绑定原理
Vue.js最核心的功能有两个,一是响应式的数据绑定系统,二是组件系统. 一.访问器属性:Object.defineProperty ECMAScript 262v5带来的新东西,FF把它归入为jav ...
- js_面向对象设计和行为委托设计模式
最近换了新工作,面试过程中有多多少少的问题没有给的出答案,为自己的技术短板而促急. javascript中万物皆对象(键:值构成的一种数据),暂且不讨论这个句话的对与错,可以想象对象在javascri ...
- linux 服务简介
Linux服务(Linux services)对于每个应用Linux的用户来说都很重要.关闭不需要的服务,可以让Linux运行的更高效,但并不是所有的Linux服务都可以关闭.今天安装了一次CentO ...
- ImportError: libQtTest.so.4: cannot open shared
错误: import cv2 File , in <module> from .cv2 import * ImportError: libQtTest.so.: cannot open s ...
- Tornado 目录
第一章:引言 1.1 Tornado是什么? 1.1.1 Tornado入门 1.1.2 社区和支持 1.2 简单的Web服务 1.2.1 Hello Tornado 1.2.1.1 参数handle ...
- Linux后台研发面试题
本系列给出了在复习过程中一些C++后台相关面试题,回答内容按照笔者的知识点掌握,故有些问题回答较为简略 1.信号的生命周期 一个完整的信号生命周期可以用四个事件刻画:1)信号诞生:2)信号在进程中注册 ...
- Linux 入门记录:十五、Linux 网络基本配置
一.以太网(Ethernet) 以太网(Ethernet)是一种计算机局域网技术.IEEE 组织的 IEEE 802.3 标准制定了以太网的技术标准,它规定了包括物理层的连线.电子信号和介质访问层协议 ...
- python基础===继承和多继承
继承 class A: def spam(self): print("A.SPAM") def add(self, x,y): return x+y class B(A): def ...