【FFT&NTT 總結】 多項式求逆 + 多項式開根

 

$FFT$總結html

(由於還不會啊,,都沒作過什麼題,因此一邊學一邊打咯。。ios

 

一、主要是用來加速卷積形式的求和吧?數組

  $F*G(n)=F[i] × G[n-i]$函數

  平時是$O(n^2)$的,FFT能夠$O(nlogn)$post

二、至關於求兩個多項式的乘積(你要求的函數是其係數)ui

  $A(x)=A0+A1*x+A2*x^2+A3*x^3+...+An−1*x^{n−1}$url

  $B(x)=B0+B1*x+B2*x^2+B3*x^3+...+Bm−1*x^{m−1}$spa

三、具體步驟?orm

  係數表達->點值表達->相乘->點值表達->係數表達htm

四、點值表示法

把多項式$A(x)$代入若干個x的值獲得若干個點$(x0,A(x0)),(x1,A(x1)),(x2,A(x2)),...,(xn−1,A(xn−1))$

 

  咱們把從點值表達式轉化爲係數表達式的操做稱爲插值

 [點值表示法對應係數表示法是有惟一性的]

五、n次單位復根

  n次單位復根是知足w^n=1的複數w,有n個。他們均勻分佈在以複平面的原點爲圓心的單位圓上

  爲$e^{\dfrac{2πki}{n}}$

  複數冪的定義$e^{ui}=cos(u)+isin(u)$

 

不如看這裏

很詳細的。

因而略過。

 

遞歸式模板

 

#include<cstdio>
#include<iostream>
#include<cmath>
#include<memory.h>
#define N 400010
using namespace std;
const double pi=acos(-1);
 
struct P
{
    double x,y;
    P() {x=y=0;}
    P(double x,double y):x(x),y(y){}
}a[N],b[N];
 
P operator + (P x,P y) {return P(x.x+y.x,x.y+y.y);}
P operator - (P x,P y) {return P(x.x-y.x,x.y-y.y);}
P operator * (P x,P y) {return P(x.x*y.x-x.y*y.y,x.x*y.y+x.y*y.x);}
 
void fft(P *s,int n,int t)
{
    if(n==1) return;
    P a0[n>>1],a1[n>>1];
    for(int i=0;i<=n;i+=2) a0[i>>1]=s[i],a1[i>>1]=s[i+1];
    fft(a0,n>>1,t);fft(a1,n>>1,t);
    P wn(cos(2*pi/n),t*sin(2*pi/n)),w(1,0);
    for(int i=0;i<(n>>1);i++,w=w*wn) s[i]=a0[i]+w*a1[i],s[i+(n>>1)]=a0[i]-w*a1[i];
    //w^2=(w+(n>>1))^2 均勻分佈在圓上面?
    //w[i^2,n]=w[i/2,n/2] 折半引理
    //s[i]=a0’(i^2)+i*a1’(i^2)=a0(i)+i*a1(i)
    //s[i+n>>1]=a0’((i+n>>1)^2)+i*a1’((i+n>>1)^2)=a0’(i^2)-i*a1’(i^2)
    //由於i=-(i+n>>1) 折半引理
}
 
int main()
{
    int n,m,nn;
    scanf("%d%d",&n,&m);
    memset(a,0,sizeof(a));memset(b,0,sizeof(b));
    for(int i=0;i<=n;i++) scanf("%lf",&a[i].x);
    for(int i=0;i<=m;i++) scanf("%lf",&b[i].x);
    nn=1;while (nn<=n+m) nn<<=1;
    fft(a,nn,1);fft(b,nn,1);
    for(int i=0;i<=nn;i++) a[i]=a[i]*b[i];
    fft(a,nn,-1);
    for(int i=0;i<=n+m;i++) printf("%d ",(int)(a[i].x/nn+0.5));
    return 0;
}

 

  

迭代式模板

高效實現$FFT$

也就是使用迭代代替遞歸

好比n=8,R數組長這樣:

$a0,a4,a2,a6,a1,a5,a3,a7$

也稱做位逆序置換

 

#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<cmath>
#include<complex>
#define Maxn 262145
#define pi acos(-1)
using namespace std;

struct P
{
    double x,y;
    P() {x=y=0;}
    P(double x,double y):x(x),y(y){}
    friend P operator + (P x,P y) {return P(x.x+y.x,x.y+y.y);}
    friend P operator - (P x,P y) {return P(x.x-y.x,x.y-y.y);}
    friend P operator * (P x,P y) {return P(x.x*y.x-x.y*y.y,x.x*y.y+x.y*y.x);}
}a[Maxn],b[Maxn];

int nn;
int R[Maxn];
void fft(P *a,int f)
{
    for(int i=0;i<nn;i++) if(i<R[i]) swap(a[i],a[R[i]]);
    for(int i=1;i<nn;i<<=1)
    {
        P wn(cos(pi/i),f*sin(pi/i));
        for(int j=0;j<nn;j+=i<<1)
        {
            P w(1,0);
            for(int k=0;k<i;k++,w=w*wn)
            {
                P x=a[j+k],y=w*a[j+k+i];
                a[j+k]=x+y;a[j+k+i]=x-y;
            }
        }
    }
}
int main()
{
    int n,m;
    scanf("%d%d",&n,&m);
    for(int i=0;i<=n;i++) scanf("%lf",&a[i].x);
    for(int i=0;i<=m;i++) scanf("%lf",&b[i].x);
    int ll=0;nn=1;
    while(nn<=n+m) ll++,nn<<=1;
    for(int i=0;i<nn;i++) R[i]=(R[i>>1]>>1)|((i&1)<<(ll-1));
    fft(a,1);fft(b,1);
    for(int i=0;i<=nn;i++) a[i]=a[i]*b[i];
    fft(a,-1);
    for(int i=0;i<=n+m;i++) printf("%d ",(int)(a[i].x/nn+0.5));
    return 0;
}

  

 


 

 

$NTT$

 

 

 

簡單複習一下原根

  一、模p下原根g,即g在模p下的階爲$\varphi(p)$

  二、g的冪構成模p下的縮系。【頗有用!

  三、p有原根當且僅當 p=2,4,質數^a,2*質數^a

  四、求原根的方法:

    暴力枚舉判斷,先找出最小的一個。

    判斷方法:對$\varphi(p)$質因數分解,假設爲$p1^{r1}*p2^{r2}...*pn^{rn}$

         有恆有 $g^{\dfrac{\varphi(p)}{pi}}!=1 (Mod P)$成立,則g是p的原根,不然不是。

    假設最小原根是$g$,則當$gcd(d,\varphi(p))==1$時,$g^d$也是模p下的原根。

    即模p下原根個數爲$\varphi(\varphi(p))$

 

對於$NTT$的題目,原根代替了n次單位復根,做用大體相同。只需把一些地方改爲Mod之類的就能夠。

 

 

具體看這裏 【這篇寫得太好了

 

 

 

 

當模數不符合要求?

 

 

具體實現方式(不完整代碼,只放主要部分)

#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
#define Maxn 500010
#define LL long long
const int Mod=998244353;
const int G=3;

int nn,R[Maxn],inv;
void ntt(int *s,int f)
{
	for(int i=0;i<nn;i++) if(i<R[i]) swap(s[i],s[R[i]]);
	for(int i=1;i<nn;i<<=1)
	{
		int wn=qpow(G,(Mod-1)/(i<<1));
		for(int j=0;j<nn;j+=(i<<1))
		{
			int w=1;
			for(int k=0;k<i;k++)
			{
				int x=s[j+k],y=1LL*w*s[j+k+i]%Mod;
				s[j+k]=(x+y)%Mod;s[j+k+i]=((x-y)%Mod+Mod)%Mod;
				w=1LL*w*wn%Mod;
			}
		}
	}
	if(f==-1)
	{
		reverse(s+1,s+nn);
		for(int i=0;i<=nn;i++) s[i]=1LL*s[i]*inv%Mod;
	}
}

int main()
{
        ///////////////////////////////
	nn=1;int ll=0;
	while(nn<=2*n) nn<<=1,ll++;
	for(int i=0;i<=nn;i++) R[i]=(R[i>>1]>>1)|((i&1)<<(ll-1));
	inv=qpow(nn,Mod-2);
	ntt(a,1);ntt(b,1);
	for(int i=0;i<=nn;i++) a[i]=1LL*a[i]*b[i]%Mod;
	ntt(a,-1);
        ///////////////////////////////
	return 0;
}

  

好啦!!!

其實有不少地方是要證實的,可是我都不會 先記住吧。。


 

多項式求逆 + 多項式開根 

 

 

 

 

 

多項式求逆的代碼實現

void get_inv(int *a,int *b,int len)  
{
    static int temp[Maxn];
    if(len==1)  
    {  
        b[0]=qpow(a[0],Mod-2);  
        b[1]=0;  
        return;  
    }  
    get_inv(a,b,len>>1);  
    memcpy(temp,a,sizeof(int)*len);  
    memset(temp+len,0,sizeof(int)*len);  
    NTT(temp,len<<1,1),NTT(b,len<<1,1);  
    for(int i=0;i<(len<<1);i++)
        b[i]=1LL*b[i]*(2-1LL*temp[i]*b[i]%Mod+Mod)%Mod;  
    NTT(b,len<<1,-1);
    memset(b+len,0,sizeof(int)*len);  
}  

 

 

len爲當前模數。不斷二分。

 

 

2017-04-14 15:00:52

相關文章
相關標籤/搜索