之前写过FFT的笔记. 我们知道FFT是在复数域上进行的变换.
而且经过数学家的证明, DFT是复数域上唯一满足循环卷积性质的变换.
而我们在OI中, 经常遇到对xxxx取模的题目, 这就启发我们可不可以在模运算的意义下找一个这样的变换.
然后我们发现有个神奇的东西, 原根\(g\), 这东西在模意义下相当于单位复根\(-e^{\frac{2\pi i}{n}}\).
所以我们预处理一下\(g\)的幂和逆元, 然后改一下fft的代码就出现了快速数论变换ntt
懒得写了 直接上代码:
void getwn(){ //预处理原根的幂和逆元
int x=qpow(3,p-2);
for(int i=0;i<20;++i){
wn[i]=qpow(3,(p-1)/(1<
inv[i]=qpow(x,(p-1)/(1<
}
}
void ntt(int *y,bool f){ rev(y); //翻转代码和fft无异
for(int m=2,id=1;m<=n;m<<=1,++id){ //id用来记录转到第几下了
for(int k=0;k
int w=1,wm=f?wn[id]:inv[id]; //如果是dft就用幂, idft就用幂的逆元
for(int j=0;j
//这里跟fft一样, 不过要对p取模
int u=y[k+j]%p,t=1ll*w*y[k+j+m/2]%p;
y[k+j]=u+t; if(y[k+j]>p) y[k+j]-=p;<