终于学会了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=0∑naixi
点值表达式就是取函数上nnn个不同的点来表示
如:多项式f(x)=x3−2x2−x+1f(x)=x^3-2x^2-x+1f(x)=x3−2x2−x+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)计算乘积
现在的问题是:由系数转点值,乘完后再转系数
然而系数转点值需要取nnn个xxx,然后求值,复杂度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-1−1就行了
定义复平面上(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,...,wnn−1
他们将平分整个复平面
折半引理
wnk=w2n2kw_n^k=w_{2n}^{2k}wnk=w2n2k
之所以叫折半而不叫倍增,主要是分数太难看所以转了一下
画个图就能证明
快速傅里叶变换
为表述方便,我们把不足的nnn增加次数凑成222的整数次幂。注意nnn为长度,最高次为n−1n-1n−1
脑子一热,不如我们把nnn个单位根作xxx吧!
开始推式子了
f(x)=∑i=0n−1aixif(x)=\sum_{i=0}^{n-1}a_ix^if(x)=i=0∑n−1aixi
闲着没事按奇偶分类
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=0∑2na2ix2i+i=0∑2n−1a2i+1x2i+1
太丑了
定义
f1(x)=∑i=0n2a2ixif_1(x)=\sum_{i=0}^{\frac{n}{2}}a_{2i}x^{i}f1(x)=i=0∑2na2ixi
f2(x)=∑i=0n2−1a2i+1xif_2(x)=\sum_{i=0}^{\frac{n}{2}-1}a_{2i+1}x^{i}f2(x)=i=0∑2n−1a2i+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}k≤2n,代入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=0n−1aiwnki
现在我们要求aaa
把bbb再插回去试试
我们倒着插单位根
即ckc_kck为多项式∑i=0n−1biwn−ki\sum_{i=0}^{n-1}b_iw_n^{-ki}i=0∑n−1biwn−ki
代入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=0∑n−1j=0∑n−1ajwnijwn−ki
把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=0∑n−1j=0∑n−1ajwnij−ki
=∑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=0∑n−1j=0∑n−1ajwni(j−k)
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=0∑n−1aji=0∑n−1wni(j−k)
考虑S=∑i=0n−1wni(j−k)S=\sum_{i=0}^{n-1}w_n^{i(j-k)}S=i=0∑n−1wni(j−k)
当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)}wnj−kS=i=1∑nwni(j−k)
(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(wnj−k−1)S=wnn(j−k)−wnj−k=wnj−k−wnj−k=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
甚至可以用来匹配字符串