BZOJ4836 [Lydsy1704月赛]二元运算 分治 多项式 FFT
原文链接http://www.cnblogs.com/zhouzhendong/p/8830036.html
题目传送门 - BZOJ4836
题意
定义二元运算$opt$满足
$$x\ opt\ y=\begin{cases}x+y & \text{$(x<y)$} \\ x-y & \text{$(x\geq y)$}\end{cases}$$
现在给定一个长为$n$的数列$a$和一个长为$m$的数列$b$,接下来有$q$次询问。每次询问给定一个数字$c$你需要求出有多少对$(i, j)$使得$a_i\ opt\ b_j=c$。
题解
这题看了标签差不多就是傻逼题QAQ。
标签:分治+FFT
好了题解就这么点QAQ。
居然写了半个小时还好1A不然完蛋。
$\vdots$
$\vdots$
$\vdots$
$\vdots$
$\vdots$
$\vdots$
$\vdots$
$\vdots$
$\vdots$
$\vdots$
$\vdots$
$\vdots$
$\vdots$
$\vdots$
$\vdots$
$\vdots$
$\vdots$
还是来正经点的吧。
首先不考虑括号里的条件,假如分别针对两种运算$x+y$和$x-y$来写,该怎么办?
$FFT$套路啊。
我们对于$a$数组搞$50001$个桶,其中下标为$i$的桶保存到就是$a_x=i$的$x$个数。
对于$b$也同理。
假装$a_i$表示原先的$a$数组中$i$的个数,$b_i$同理。
然后你要快速算$x+y$相同的,写下式子:
$$h_i=\sum_{j=0}^{i}a_jb_{i-j}$$
显然可以$FFT$优化。
然后你要算$x-y$相同的,你假装桶有负的下标,对于每一个$i$,使$b_{-i}=b_i$。
然后让$b$整体向下标的正方向移动$50000$个单位(对于有效的前$50001$个,这个操作的效果就是将原来的$50001$个桶翻转),就可以列出和原来的那个差不多的式子,同样也可以$FFT$。
但是原运算法则中有括号里面的条件啊。
我们要强力满足这个条件。
于是我们采用分治。
对数值进行分治。
我们分$3$类对最终答案进行贡献。
假装我们把区间分成了$[L,mid]$和$(mid,R]$。
$1.x=y$,只要再长度为$1$的区间内自己贡献一下就可以了。具体看代码。
$2.x<y$,那我们只要让$a[L,mid]$和$b(mid,R]$按照原式的第一种运算方式算,并$FFT$优化即可。
$3.x>y$,那我们只要让$a(mid,R]$和$b[l,mid]$按照原式的第二种运算方式算,并$FFT$优化即可。
伪代码:
solve(L,R)
if (L=R)
-->1.x=y
return
-->2.x<y
-->3.x>y
solve(L,mid),solve(mid+1,R)
代码
#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
const int N=1<<17;
double PI=acos(-1.0);
int read(){
int x=0;
char ch=getchar();
while (!('0'<=ch&&ch<='9'))
ch=getchar();
while ('0'<=ch&&ch<='9')
x=(x<<3)+(x<<1)+ch-48,ch=getchar();
return x;
}
int T,n,m,q,a[N],b[N];
LL tot[N];
int s,d,Rev[N];
struct C{
double r,i;
C(){}
C(double a,double b){r=a,i=b;}
C operator + (C x){return C(r+x.r,i+x.i);}
C operator - (C x){return C(r-x.r,i-x.i);}
C operator * (C x){return C(r*x.r-i*x.i,r*x.i+i*x.r);}
}w[N],A[N],B[N];
void FFT(C a[],int n){
for (int i=0;i<n;i++)
if (i<Rev[i])
swap(a[i],a[Rev[i]]);
for (int t=n>>1,d=1;d<n;d<<=1,t>>=1)
for (int i=0;i<n;i+=(d<<1))
for (int j=0;j<d;j++){
C tmp=w[t*j]*a[i+j+d];
a[i+j+d]=a[i+j]-tmp;
a[i+j]=a[i+j]+tmp;
}
}
void solve(int L,int R){
if (L==R){
tot[0]+=1LL*a[L]*b[R];
return;
}
int mid=(L+R)>>1;
//a[L]...a[mid] VS b[mid+1]...b[R]
for (s=1,d=0;s<R-L;s<<=1,d++);
for (int i=0;i<s;i++){
Rev[i]=(Rev[i>>1]>>1)|((i&1)<<(d-1));
w[i]=C(cos(2*i*PI/s),sin(2*i*PI/s));
A[i]=B[i]=C(0,0);
}
for (int i=L;i<=mid;i++)
A[i-L]=C(a[i],0);
for (int i=mid+1;i<=R;i++)
B[i-mid-1]=C(b[i],0);
FFT(A,s),FFT(B,s);
for (int i=0;i<s;i++)
A[i]=A[i]*B[i],w[i].i*=-1.0;
FFT(A,s);
for (int i=0;i<R-L;i++)
tot[i+L+mid+1]+=(LL)(A[i].r/s+0.5);
//a[mid+1]...a[R] VS b[L]...b[mid]
for (int i=0;i<s;i++)
A[i]=B[i]=C(0,0),w[i].i*=-1.0;
for (int i=mid+1;i<=R;i++)
A[i-mid-1]=C(a[i],0);
for (int i=L;i<=mid;i++)
B[mid-i]=C(b[i],0);
FFT(A,s),FFT(B,s);
for (int i=0;i<s;i++)
A[i]=A[i]*B[i],w[i].i*=-1.0;
FFT(A,s);
for (int i=0;i<R-L;i++)
tot[i+mid+1-mid]+=(LL)(A[i].r/s+0.5);
solve(L,mid),solve(mid+1,R);
}
int main(){
T=read();
while (T--){
n=read(),m=read(),q=read();
memset(a,0,sizeof a);
memset(b,0,sizeof b);
while (n--)
a[read()]++;
while (m--)
b[read()]++;
memset(tot,0,sizeof tot);
solve(0,50000);
while (q--)
printf("%lld\n",tot[read()]);
}
return 0;
}
BZOJ4836 [Lydsy1704月赛]二元运算 分治 多项式 FFT的更多相关文章
- BZOJ4836: [Lydsy1704月赛]二元运算
BZOJ4836: [Lydsy1704月赛]二元运算 https://lydsy.com/JudgeOnline/problem.php?id=4836 分析: 分开做,维护两个桶. 分治每次求\( ...
- bzoj 4836 [Lydsy1704月赛]二元运算 分治FFT+生成函数
[Lydsy1704月赛]二元运算 Time Limit: 8 Sec Memory Limit: 128 MBSubmit: 577 Solved: 201[Submit][Status][Di ...
- BZOJ4836: [Lydsy1704月赛]二元运算【分治FFT】【卡常(没卡过)】
Description 定义二元运算 opt 满足 现在给定一个长为 n 的数列 a 和一个长为 m 的数列 b ,接下来有 q 次询问.每次询问给定一个数字 c 你需要求出有多少对 (i, j) 使 ...
- BZOJ 4836: [Lydsy1704月赛]二元运算 分治FFT
Code: #include<bits/stdc++.h> #define ll long long #define maxn 500000 #define setIO(s) freope ...
- bzoj 4836: [Lydsy2017年4月月赛]二元运算 -- 分治+FFT
4836: [Lydsy2017年4月月赛]二元运算 Time Limit: 8 Sec Memory Limit: 128 MB Description 定义二元运算 opt 满足 现在给定一 ...
- 【bzoj4836】[Lydsy2017年4月月赛]二元运算 分治+FFT
题目描述 定义二元运算 opt 满足 现在给定一个长为 n 的数列 a 和一个长为 m 的数列 b ,接下来有 q 次询问.每次询问给定一个数字 c 你需要求出有多少对 (i, j) 使得 a_ ...
- CodeForces 958F3 Lightsabers (hard) 启发式合并/分治 多项式 FFT
原文链接http://www.cnblogs.com/zhouzhendong/p/8835443.html 题目传送门 - CodeForces 958F3 题意 有$n$个球,球有$m$种颜色,分 ...
- UOJ#23. 【UR #1】跳蚤国王下江南 仙人掌 Tarjan 点双 圆方树 点分治 多项式 FFT
原文链接https://www.cnblogs.com/zhouzhendong/p/UOJ23.html 题目传送门 - UOJ#23 题意 给定一个有 n 个节点的仙人掌(可能有重边). 对于所有 ...
- BZOJ3451 Tyvj1953 Normal 点分治 多项式 FFT
原文链接https://www.cnblogs.com/zhouzhendong/p/BZOJ3451.html 题目传送门 - BZOJ3451 题意 给定一棵有 $n$ 个节点的树,在树上随机点分 ...
随机推荐
- VMware虚拟机无法上网 无法启动VMnet0等问题【转载】
http://www.linuxidc.com/Linux/2015-05/117704.htm VMware虚拟机无法上网,由于之前安装过VMware虚拟机,后来将它卸载了,然后重新安装,最后出现了 ...
- Eclipse 添加 Source 源代码、Javadoc 文档
源代码 Source 按住 Ctrl 键,鼠标放到对应的类.方法上,出现 Open Declaration,Open Implementation ,可查看对应的实现.声明源代码. 也可以在[Proj ...
- CMDB服务器管理系统【s5day89】:采集资产之汇报信息
1.服务器端收到的数据和客户端的数据不一样 print(request.post) 少发了,还是少取了,说明根本没有把数据全发过来 print(request.body) 1.只把字典的key给我发过 ...
- Matlab怎么修改显示数值格式/精度/小数位数
参考:https://jingyan.baidu.com/article/7f41ecec1ad029593c095c70.html
- Jenkins-在windows上配置自动化部署(Jenkins+Bonobo.Git.Server)
本文配置Jenkins.git服务器采用 Bonobo.Git.Server 1. 登录后,打开Jenkins界面,新建一个任务 2. 配置信息 3. 配置git项目地址,我们先进行其他配置,等会再继 ...
- VMware 虚拟机运行卡慢的解决办法
虚拟机的运行速度有一部分原因决定于你的物理机的配置,如果你的物理机配置不高,建议不要运行虚拟机,那样反而会更卡! 切记!!!在操作之前,先给虚拟机拍摄一个快照,然后所有操作都在虚拟机关机的情况下执行! ...
- 微星X470主板装机
记录一下装机过程,以作纪念 配置 机箱:先马黑洞3 电源:先马金牌500w CPU:AMD 锐龙5:2600X 主板:微星 X470 暗黑版 显卡:七彩虹 RTX2060 内存:科赋 3200,2条8 ...
- 无连接运输的UDP、可靠数据传输原理、面向连接运输的TCP
由[RFC 768]定义的UDP只是做了运输协议能够做的最少工作.除了复用/分解功能极少量的差错检测外,它几乎没有对IP增加别的东西.如果应用程序开发人员选择UDP而不是TCP,则该应用程序差不多就是 ...
- 第五节:SignalR大杂烩(与MVC融合、全局的几个配置、跨域的应用、C/S程序充当Client和Server)
一. 说在前面的话 本节主要在前面章节的基础上补充了几个简单的知识点,比如:第三方调用通过 GlobalHost.ConnectionManager.GetHubContext<MySpecHu ...
- Gram 矩阵与向量到子空间的距离
设 $W$ 是 $n$ 维 Euclidean 空间 $V$ 的子空间, $\beta\in V$, 定义 $\beta$ 到 $W$ 的距离 $$\bex \rd (\beta,W)=|\bet ...