FFT:从入门到沉迷

终于学会了FFTFFTFFT并无法自拔

啥是FFT

FFTFFTFFT指快速傅里叶变换

OIOIOI中的应用是O(nlogn)O(nlogn)O(nlogn)计算函数卷积

人话:多项式乘法

多项式毒瘤模板题的万恶之源

系数表达式与点值表达式

系数表达式就是平常的表示方法

f(x)=∑i=0naixif(x)=\sum_{i=0}^{n}a_{i}x^if(x)=i=0naixi

点值表达式就是取函数上nnn个不同的点来表示

如:多项式f(x)=x3−2x2−x+1f(x)=x^3-2x^2-x+1f(x)=x32x2x+1

点值表达式可以为:(−1,−1)(0,1)(1,−1)(-1,-1)(0,1)(1,-1)(1,1)(0,1)(1,1)

由代数基本定理,两者等价

初步推导

为什么要引入点值表达式?

如果相乘的多项式选取了相同的xxx,就可以O(n)O(n)O(n)计算乘积

现在的问题是:由系数转点值,乘完后再转系数

然而系数转点值需要取nnnxxx,然后求值,复杂度O(n2)O(n^2)O(n2)

更尴尬的是点值转系数高斯消元O(n3)O(n^3)O(n3)

看来要在取数上做手脚了

复平面与单位根

定义i2=−1i^2=-1i2=1,则a+bia+bia+bi表示一个复数

不需要纠结iii是个啥,只需要保留,遇到i2i^2i2就化成−1-11就行了

定义复平面上(a,b)(a,b)(a,b)表示a+bia+bia+bi,也可看成向量

有个奇妙的性质:复数相乘,相当于转角(xxx正半轴逆时针旋转)相加,长度相乘

定义nnn的单位根wnw_nwn满足wnn=1w_n^n=1wnn=1

这样的单位根有nnn个,为wn0,wn1,wn2,...,wnn−1w_n^0,w_n^1,w_n^2,...,w_n^{n-1}wn0,wn1,wn2,...,wnn1

他们将平分整个复平面

折半引理

wnk=w2n2kw_n^k=w_{2n}^{2k}wnk=w2n2k

之所以叫折半而不叫倍增,主要是分数太难看所以转了一下

画个图就能证明

快速傅里叶变换

为表述方便,我们把不足的nnn增加次数凑成222的整数次幂。注意nnn为长度,最高次为n−1n-1n1

脑子一热,不如我们把nnn个单位根作xxx吧!

开始推式子了

f(x)=∑i=0n−1aixif(x)=\sum_{i=0}^{n-1}a_ix^if(x)=i=0n1aixi

闲着没事按奇偶分类

f(x)=∑i=0n2a2ix2i+∑i=0n2−1a2i+1x2i+1f(x)=\sum_{i=0}^{\frac{n}{2}}a_{2i}x^{2i}+\sum_{i=0}^{\frac{n}{2}-1}a_{2i+1}x^{2i+1}f(x)=i=02na2ix2i+i=02n1a2i+1x2i+1

太丑了

定义

f1(x)=∑i=0n2a2ixif_1(x)=\sum_{i=0}^{\frac{n}{2}}a_{2i}x^{i}f1(x)=i=02na2ixi

f2(x)=∑i=0n2−1a2i+1xif_2(x)=\sum_{i=0}^{\frac{n}{2}-1}a_{2i+1}x^{i}f2(x)=i=02n1a2i+1xi

那么有

f(x)=f1(x2)+xf2(x2)f(x)=f_1(x^2)+xf_2(x^2)f(x)=f1(x2)+xf2(x2)

对于k≤n2k\leq\frac{n}{2}k2n,代入x=wnkx=w_n^kx=wnk

f(wnk)=f1(wn2k)+wnkf2(wn2k)f(w_n^k)=f_1(w_n^{2k})+w_n^kf_2(w_n^{2k})f(wnk)=f1(wn2k)+wnkf2(wn2k)

同时

f(wnk+n2)=f1(wn2k+n)+wnk+n2f2(wn2k+n)f(w_n^{k+\frac{n}{2}})=f_1(w_n^{2k+n})+w_n^{k+\frac{n}{2}}f_2(w_n^{2k+n})f(wnk+2n)=f1(wn2k+n)+wnk+2nf2(wn2k+n)

显然wnn=1,w2nn=−1w_n^n=1,w_{2n}^{n}=-1wnn=1,w2nn=1

所以

f(wnk+n2)=f1(wn2k)−wnkf2(wn2k)f(w_n^{k+\frac{n}{2}})=f_1(w_n^{2k})-w_n^kf_2(w_n^{2k})f(wnk+2n)=f1(wn2k)wnkf2(wn2k)

我们发现两个式子结构是相同的

根据折半引理

f1(wn2k)=f1(wn2k)f_1(w_n^{2k})=f_1(w_{\frac{n}{2}}^{k})f1(wn2k)=f1(w2nk)

f2(wn2k)=f2(wn2k)f_2(w_n^{2k})=f_2(w_{\frac{n}{2}}^{k})f2(wn2k)=f2(w2nk)

f(wnk)=f1(wn2k)+wnkf2(wn2k)f(w_n^k)=f_1(w_{\frac{n}{2}}^{k})+w_n^kf_2(w_{\frac{n}{2}}^{k})f(wnk)=f1(w2nk)+wnkf2(w2nk)

f(wnk+n2)=f1(wn2k)−wnkf2(wn2k)f(w_n^{k+\frac{n}{2}})=f_1(w_{\frac{n}{2}}^{k})-w_n^kf_2(w_{\frac{n}{2}}^{k})f(wnk+2n)=f1(w2nk)wnkf2(w2nk)

我们发现规模不断减半,然后可以愉快的递归了。

复杂度O(NlogN)O(NlogN)O(NlogN)

逆变换

现在我们转成了点值,进行了乘法,该转回系数了。

假设原多项式系数是aaa,将NNN个单位根代入得到了NNN个复数bbb

即:bk=∑i=0n−1aiwnkib_k=\sum_{i=0}^{n-1}a_iw_n^{ki}bk=i=0n1aiwnki

现在我们要求aaa

bbb再插回去试试

我们倒着插单位根

ckc_kck为多项式∑i=0n−1biwn−ki\sum_{i=0}^{n-1}b_iw_n^{-ki}i=0n1biwnki

代入bbb

ck=∑i=0n−1∑j=0n−1ajwnijwn−kic_k=\sum_{i=0}^{n-1}\sum_{j=0}^{n-1}a_jw_n^{ij}w_n^{-ki}ck=i=0n1j=0n1ajwnijwnki

wnw_nwn合并

ck=∑i=0n−1∑j=0n−1ajwnij−kic_k=\sum_{i=0}^{n-1}\sum_{j=0}^{n-1}a_jw_n^{ij-ki}ck=i=0n1j=0n1ajwnijki

=∑i=0n−1∑j=0n−1ajwni(j−k)\quad=\sum_{i=0}^{n-1}\sum_{j=0}^{n-1}a_jw_n^{i(j-k)}=i=0n1j=0n1ajwni(jk)

aja_jaj可以放前面来

ck=∑j=0n−1aj∑i=0n−1wni(j−k)c_k=\sum_{j=0}^{n-1}a_j\sum_{i=0}^{n-1}w_n^{i(j-k)}ck=j=0n1aji=0n1wni(jk)

考虑S=∑i=0n−1wni(j−k)S=\sum_{i=0}^{n-1}w_n^{i(j-k)}S=i=0n1wni(jk)

j=kj=kj=k,显然S=nS=nS=n

j≠kj \neq kj̸=k

wnj−kS=∑i=1nwni(j−k)w_n^{j-k}S=\sum_{i=1}^{n}w_n^{i(j-k)}wnjkS=i=1nwni(jk)

(wnj−k−1)S=wnn(j−k)−wnj−k=wnj−k−wnj−k=0(w_n^{j-k}-1)S=w_n^{n(j-k)}-w_n^{j-k}=w_n^{j-k}-w_n^{j-k}=0(wnjk1)S=wnn(jk)wnjk=wnjkwnjk=0

所以S=0S=0S=0

这说明只有j=kj=kj=k时对答案有贡献

此时

ck=aknc_k=a_knck=akn

ak=ckna_k=\frac{c_k}{n}ak=nck

FFTFFTFFT至此就结束了。

代码:

#include <iostream>
#include <cstdio>
#include <cmath>
#include <cctype>
#define MAXN 2000005
const double PI=acos(-1.0);
using namespace std;
inline int read()
{int ans=0,f=1;char c=getchar();while (!isdigit(c)){if (c=='-')f=-1;c=getchar();}while (isdigit(c)){ans=(ans<<3)+(ans<<1)+(c^48);c=getchar();}return f*ans;
}
struct complex
{double x,y;complex(double x=0,double y=0):x(x),y(y){}
}a[MAXN],b[MAXN];
complex operator +(const complex& a,const complex& b){return complex(a.x+b.x,a.y+b.y);}
complex operator -(const complex& a,const complex& b){return complex(a.x-b.x,a.y-b.y);}
complex operator *(const complex& a,const complex& b){return complex(a.x*b.x-a.y*b.y,a.x*b.y+a.y*b.x);}
void fft(int lim,complex *a,int type)
{if (lim==1)return;complex a1[lim>>1],a2[lim>>1];for (int i=0;i<=lim;i+=2)a1[i>>1]=a[i],a2[i>>1]=a[i+1];fft(lim>>1,a1,type);fft(lim>>1,a2,type);complex wn(cos(2.0*PI/lim),type*sin(2.0*PI/lim)),w(1,0);for (int i=0;i<(lim>>1);i++,w=w*wn){a[i]=a1[i]+w*a2[i];a[i+(lim>>1)]=a1[i]-w*a2[i];}
}
int main()
{int n,m;n=read(),m=read();for (int i=0;i<=n;i++)//注意下标 a[i].x=read();for (int i=0;i<=m;i++)b[i].x=read();int lim=1;while (lim<=n+m)lim<<=1;fft(lim,a,1);fft(lim,b,1);for (int i=0;i<=lim;i++)	a[i]=a[i]*b[i];fft(lim,a,-1);for (int i=0;i<=n+m;i++)printf("%d ",(int)(a[i].x/lim+0.5));return 0;
}

迭代实现

什么?你没过?

观察一下原来的代码 我们发现数组开小了

在递归函数中,每一层都会开一个数组,这造成效率极低。

我们观察递归的全过程

(二进制)

000 001 010 011 100 101 110 111
000 010 100 110|001 011 101 111
000 100|010 110|001 101|011|111

我们发现,递归完后实际上是将二进制翻转

那我们可以模拟递归的过程,算出最后的下标,再一层一层推回去

如何计算翻转后的数?

①将第二位及以上的翻转并右移1位

②将第一位移到最高位

可以O(N)O(N)O(N)求出

#include <iostream>
#include <cstdio>
#include <cstring>
#include <cctype>
#include <cmath>
#define MAXN 5000005
const double Pi=acos(-1.0);
using namespace std;
inline int read()
{int ans=0,f=1;char c=getchar();while (!isdigit(c)){if (c=='-') f=-1;c=getchar();}while (isdigit(c)) ans=(ans<<3)+(ans<<1)+(c^48),c=getchar();return f*ans;
}
struct complex{double x,y;complex(const double& x=0,const double& y=0):x(x),y(y){}}a[MAXN],b[MAXN];
inline complex operator +(const complex& a,const complex& b){return complex(a.x+b.x,a.y+b.y);}
inline complex operator -(const complex& a,const complex& b){return complex(a.x-b.x,a.y-b.y);}
inline complex operator *(const complex& a,const complex& b){return complex(a.x*b.x-a.y*b.y,a.x*b.y+a.y*b.x);}
int lim=1,l,r[MAXN];
void fft(complex* a,int type)
{for (int i=0;i<lim;i++) if (i<r[i]) swap(a[i],a[r[i]]);// 计算最后的序列。<是为了避免换回来,写>也一样for (int len=2;len<=lim;len<<=1)//合并出的长度{int mid=(len>>1);//被合并的长度complex Wn(cos(Pi/mid),type*sin(Pi/mid));//单位根for (int s=0;s<=lim;s+=len)//枚举每一段的长度{complex w(1,0);for (int k=0;k<mid;k++,w=w*Wn){complex x=a[s+k],y=w*a[s+mid+k];//一个小优化,避免大量复数乘法运算a[s+k]=x+y,a[s+mid+k]=x-y;}}}
}
int main()
{int n,m;n=read(),m=read();for (int i=0;i<=n;i++) a[i].x=read();for (int i=0;i<=m;i++) b[i].x=read();while (lim<=n+m) lim<<=1,++l;for (int i=0;i<lim;i++) r[i]=(r[i>>1]>>1)|((i&1)<<(l-1));//r[i]表示i翻转后的数fft(a,1),fft(b,1);for (int i=0;i<=lim;i++) a[i]=a[i]*b[i];fft(a,-1);for (int i=0;i<=n+m;i++) printf("%d ",(int)(a[i].x/lim+0.5));return 0;
}

凡是两个数组一一对应然后进行各种运算的,都可以考虑FFTFFTFFT

甚至可以用来匹配字符串

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/315295.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

HDU - 6769-In Search of Gold-二分+树形dp

https://vjudge.net/problem/HDU-6769 题目大意&#xff1a;给你n个点&#xff0c;有n-1条边&#xff0c;每条边有a&#xff0c;b两个权值&#xff0c;给你一个k&#xff0c;恰好有k条边的权值取a&#xff0c;其余取b的时候&#xff0c;树的直径的最小值。 思路&#xff1a;答…

Codeforces Round #701 (Div. 2) E. Move and Swap 思维 + dp

传送门 文章目录题意&#xff1a;思路&#xff1a;题意&#xff1a; 思路&#xff1a; 由于是按层来的&#xff0c;所以我们肯定先按照层来分组。 定义dp[i]dp[i]dp[i]为红棋在位置iii的时候的最大得分和。 先考虑不换的情况&#xff0c;我们对于每个点都从他的父节点转移过来…

用 docker-compose 启动 WebApi 和 SQL Server

本系列文章所要做出的演示架构基于 .NET Core Web Api、MSSQL、Skywalking 和 nginx &#xff0c;这些都会通过docker-compose一键创建/启动容器&#xff0c;然后用 Azure DevOps 发布上线。所以本系列文章重点并不是如何写好.NET Core&#xff0c;而是围绕着 .NET Core 的容器…

OI群论:从入门到自闭

怎么这么多阅读啊…… 这篇文章是群论&#xff08;其实只有Polya&#xff09;在信息学奥赛中计数的运用&#xff0c;并不是群论的讲义…… 群论可以说是数学中最高深的内容&#xff0c;一个oier去深入了解是不现实的。因此&#xff0c;我们只需要知道结论。 本文主要讲解群论…

2020牛客暑期多校训练营(第二场)Just Shuffle

https://ac.nowcoder.com/acm/contest/5667/J 题目大意&#xff1a;给你一个置换A&#xff0c;使得置换P^kA,让你求出置换P。 思路&#xff1a;我们根据置换A再置换z次&#xff0c;那么就等于置换p 置换z*k次,如果z*k%len0&#xff0c;那么将会回到单位序列&#xff0c;那我们…

Codeforces Round #619 (Div. 2) F. Super Jaber 多源bfs + 思维转换

传送门 文章目录题意&#xff1a;思路&#xff1a;题意&#xff1a; 给你一个矩阵&#xff0c;每个格子都有一个颜色kkk&#xff0c;每秒可以移动到相邻矩阵或者瞬移到同一颜色的任意矩阵。有qqq个询问&#xff0c;每次询问给出两个点&#xff0c;求从一个点到另一个点的最短时…

你可以保持沉默,但你所说的一切都将成为呈堂证供——浅谈Azure WORM保护

本文作者|Yuan Han本文来源|Reid爸的菜园子美国安然事件后&#xff0c;电子数据的合规性保存越来越受到重视&#xff1b;各国政府制定了一系列的法律&#xff0c;如美国《赛班斯法案》等&#xff0c;对于不同类型的电子数据保留期限做了严格规定&#xff1b;国内也没落后&#…

后缀自动机:从入门到放弃

写在前面 后缀自动机&#xff0c;简称SAMSAMSAM,是一种十分优秀的字符串匹(shu)配(ju)算(jie)法(gou) 字符串界的bossbossboss&#xff0c;几乎可以解决全部正常的字符串题目 至少我前前后后学了一年&#xff0c;听过444次课&#xff0c;几度怀疑自己不适合oioioi 请做好心…

2021牛客第一场 K.Knowledge Test about Match

https://ac.nowcoder.com/acm/contest/11166/K 题意就是使得图中的那个式子最小&#xff0c;你的答案不一定是要最标准的&#xff0c;只要平均水平下和标准值的偏差不超过4%就行了。 有了这个提示&#xff0c;那我们直接贪心瞎搞就行了&#xff0c;只有符合换过去的收益的增大…

Codeforces Round #620 (Div. 2) F2. Animal Observation (hard version) dp + 线段树

传送门 文章目录题意&#xff1a;思路&#xff1a;题意&#xff1a; 比如下面这个图&#xff1a; 思路&#xff1a; 对于这个题&#xff0c;比较容易就能考虑到dpdpdp&#xff0c;设f[i][j]f[i][j]f[i][j]为到了第iii行&#xff0c;覆盖了[j,jk−1][j,jk-1][j,jk−1]范围时候…

设计模式之总体介绍

1. 背景与介绍设计模式是经过反复使用、经过分类的代码总结。设计模式的目的是提高代码可重用性和可靠性&#xff0c;并使代码条理清晰、易于理解、易于维护。设计模式描述了在各种情况下&#xff0c;要选择什么样的方案来解决问题。设计模式通常以类和对象来描述其中的关系和相…

回文自动机:从入门到只会打板

写在前面 如果你会SAMSAMSAM&#xff0c;相信回文自动机不会难懂。 如果你不会&#xff0c;你可以参考我的上一篇文章。 至少回文自动机是治愈系的吧。 作用 回文自动机&#xff0c;也叫回文树&#xff0c;简称PAMPAMPAM实际上它既不是自动机也不是树 处理回文串的有力工…

2021牛客第一场 I. Increasing Subsequence-前缀和优化dp

https://ac.nowcoder.com/acm/contest/11166/I 思路&#xff1a;dp[i][j] 是表示上上步走在i点&#xff0c;上一步走在j点的期望。首先我们很容易想到n^3的做法&#xff0c;那我们必须考虑去优化一维的时间复杂度。我们可以考虑使用前缀和优化dp转移。 我们枚举i点&#xff0c…

Codeforces Round #620 (Div. 2) E. 1-Trees and Queries 思维 + LCA

传送门 文章目录题意思路&#xff1a;题意 思路&#xff1a; 照例&#xff0c;先考虑不加边怎么做。由于可以经过重复的边或点&#xff0c;设aaa与bbb之间长度为lenlenlen&#xff0c;那么需要len<klen<klen<k并且还需要(k−len)mod20(k-len) \bmod 20(k−len)mod20&…

.NET Core 微服务之Polly熔断策略

紧接着上一篇说&#xff0c;咱们继续介绍Polly这个类库熔断策略&#xff08;Circuit-breaker&#xff09;如果调用某个目标服务出现过多超时、异常等情况&#xff0c;可以采取一定时间内熔断该服务的调用&#xff0c;熔断期间的请求将不再继续调用目标服务&#xff0c;而是直接…

【洛谷P4169】天使玩偶/SJY摆棋子【CDQ分治】

传送门 题意&#xff1a;动态加点&#xff0c;给定点询问曼哈顿距离最近的点 N,M≤3e5,x,y≤1e6N,M \leq 3e5,x,y \leq 1e6N,M≤3e5,x,y≤1e6 经(kan)过(le)分(ti)析(jie),这是一道cdqcdqcdq分治 考虑当前区间左半边修改对右半边的询问的影响 设左边某个修改为(x1,y1)(x_1,…

牛客第二场 G.League of Legends-单调队列优化dp

https://ac.nowcoder.com/acm/contest/11253/G 上面出题人给的题解&#xff1a; 思路基本差不多&#xff0c;这里主要说一下合并小区间的dp&#xff0c; dp[i][j]代表前i个分成j组最大的时间max 我们首先将区间排好序&#xff0c;如果满足a[k]>b[i] ,则有 j都是由j-1转…

.NET中扩展方法和Enumerable(System.Linq)

LINQ是我最喜欢的功能之一&#xff0c;程序中到处是data.Where(xx>5).Select(x)等等的代码&#xff0c;她使代码看起来更好&#xff0c;更容易编写&#xff0c;使用起来也超级方便&#xff0c;foreach使循环更加容易&#xff0c;而不用for int..&#xff0c;linq用起来那么爽…

Peaks加强版 黑暗爆炸 - 3551 Kruskal重构树 + 主席树

传送门 文章目录题意&#xff1a;思路&#xff1a;题意&#xff1a; 给你一张图&#xff0c;有nnn个山峰&#xff0c;每个山峰高度为hih_ihi​&#xff0c;有mmm条边&#xff0c;每条边有个难度值wiw_iwi​&#xff0c;现在有qqq个询问&#xff0c;每次询问给定一个山峰vvv&am…

【BJOI2017】树的难题【点分治】【线段树】

传送门 传送门 题意&#xff1a;给一棵树&#xff0c;树上有颜色&#xff0c;每种颜色有权值&#xff0c;定义一条路径的权值为所有颜色相同段的权值之和&#xff0c;求长度在[L,R][L,R][L,R]中的路径的最大权值。 数据范围&#xff1a;暴力过不了 显然是个点分治 对于分治…