zoukankan      html  css  js  c++  java
  • 快速傅里叶变换(FFT)

    快速傅里叶变换,简称FFT,是一种可以O(nlogn)的时间内计算n次多项式乘法的算法。

    写得很好的博客:自为风月马前卒大佬的博客

    大致步骤是:

    1.先将两个多项式的系数表达法O(nlogn)都转化成点值表达法。

    例如:y=A0+A1*x+A2*x2+A3*x3+...+An*xn可以转化为n+1个点(x1,y1),(x2,y2),(x3,y3),...,(xn,yn),(xn+1,yn+1)

    这n+1个点也能确定这个多项式,就像两点确定一条直线,三点确定一条抛物线一样。

    同理第二个多项式也转化为n+1个点,这n+1个点的x要与第一个多项式的点的x分别相等。

    2.x相同的点所对应的y可以O(n)直接相乘,达到多项式相乘的效果。

    3.再将点值表达法转化为系数表达法就行了。

    然后发现暴力转化的话时间是O(n2),那么如何优化呢,这里就要牵扯到复数的概念和性质了,前面的博客里写的非常好,我就不再赘述了其实就是懒

    下面是递归版的代码和非递归版的代码:(递归版的时间又长空间又大好像没什么用,但应该可以帮助理解吧w)

    #include<bits/stdc++.h>
    using namespace std;
    const int N=4e6+1e5;
    const double pai=acos(-1.0);
    double Co[N],Si[N];
    struct Complex{
        double x,y;
        Complex(double xx=0,double yy=0){
            x=xx; y=yy;
        }
    }A[N],B[N];
    Complex operator+(Complex X,Complex Y) { return Complex(X.x+Y.x,X.y+Y.y); }
    Complex operator-(Complex X,Complex Y) { return Complex(X.x-Y.x,X.y-Y.y); }
    Complex operator*(Complex X,Complex Y) { return Complex(X.x*Y.x-X.y*Y.y,X.x*Y.y+X.y*Y.x); }
    int rd(){
        int s=0,ff=1;
        char w=getchar();
        while(!isdigit(w))
            ff^=w=='-',w=getchar();
        while(isdigit(w))
            s=s*10+w-'0',w=getchar();
        return ff?s:-s;
    }
    void FFt(int n,Complex *a,int Opt){
        if(n==1) return ;
        Complex c[n>>1],d[n>>1],k;
        for(int i=0;i<n;i+=2)//注意这里没有等于号 
            c[i>>1]=a[i],d[i>>1]=a[i+1];
        FFt(n>>1,c,Opt); FFt(n>>1,d,Opt);
        Complex w=Complex(Co[n],Opt*Si[n]),x=Complex(1,0); n>>=1;
        for(int i=0;i<n;i++,x=x*w)
            k=x*d[i],a[i]=c[i]+k,a[i+n]=c[i]-k;
    }
    int main(){
        int n=1,N,M;
        N=rd(); M=rd();
        while(n<=N+M) n<<=1;
        for(int i=1;i<=n;i++)
            Co[i]=cos(2.0*pai/i),Si[i]=sin(2.0*pai/i);
        for(int i=0;i<=N;i++) A[i].x=rd();
        for(int i=0;i<=M;i++) B[i].x=rd();
        FFt(n,A,1); FFt(n,B,1);
        for(int i=0;i<=n;i++) A[i]=A[i]*B[i];
        FFt(n,A,-1);
        for(int i=0;i<=N+M;i++)
            printf("%d ",(int)(A[i].x/n+0.5));
        return 0;
    }
    递归版
    #include<bits/stdc++.h>
    using namespace std;
    const int N=4e6+1e5;
    const double pai=acos(-1.0);
    int r[N];
    double Co[N],Si[N];
    struct Complex{
        double x,y;
        Complex(double xx=0,double yy=0){
            x=xx; y=yy;
        }
    }A[N],B[N];
    Complex operator+(Complex X,Complex Y) { return Complex(X.x+Y.x,X.y+Y.y); }
    Complex operator-(Complex X,Complex Y) { return Complex(X.x-Y.x,X.y-Y.y); }
    Complex operator*(Complex X,Complex Y) { return Complex(X.x*Y.x-X.y*Y.y,X.x*Y.y+X.y*Y.x); }
    int rd(){
        int s=0,ff=1;
        char w=getchar();
        while(!isdigit(w))
            ff^=w=='-',w=getchar();
        while(isdigit(w))
            s=s*10+w-'0',w=getchar();
        return ff?s:-s;
    }
    void FFt(int n,Complex *a,int Opt){
        Complex x,w,X,Y;
        for(int i=0;i<n;i++) //注意无等于号 
            if(i<r[i]) swap(a[i],a[r[i]]);
        for(int mid=1;mid<n;mid<<=1){
            for(int l=0,j=(mid<<1);l<n;l+=j){
                w=Complex(Co[mid<<1],Opt*Si[mid<<1]); x=Complex(1,0);
                for(int i=l;i<l+mid;i++,x=x*w){
                    X=a[i],Y=x*a[i+mid];
                    a[i]=X+Y,a[i+mid]=X-Y;
                }
            }
        }
    }
    int main(){
        int n=1,N,M,l=0; N=rd(); M=rd();
        while(n<=N+M) n<<=1,l++;
        for(int i=1;i<n;i++)
            r[i]=((r[i>>1]>>1)|((i&1)<<(l-1)));
        for(int i=1;i<=n;i++)
            Co[i]=cos(2.0*pai/i),Si[i]=sin(2.0*pai/i);
        for(int i=0;i<=N;i++) A[i].x=rd();
        for(int i=0;i<=M;i++) B[i].x=rd();
        FFt(n,A,1); FFt(n,B,1);
        for(int i=0;i<=n;i++) A[i]=A[i]*B[i];
        FFt(n,A,-1);
        for(int i=0;i<=N+M;i++)
            printf("%d ",(int)(A[i].x/n+0.5));
        return 0;
    }
    非递归版
  • 相关阅读:
    scp 指定端口(转)
    openshift 入门 部署 openshift-origin-server-v3.7.0
    kubernetes 网络模型
    故障排除--kubernetes 运维操作步骤 -- kubedns -- busybox -- nslookup 问题
    Service 服务发现的两种方式-通过案例来理解+服务外部访问类型+selector-label
    nmap 扫描端口 + iftop 实时监控流量
    Intellij IDEA 2016.3.4 注册激活--转
    laravel服务提供者类说明
    使用PHP实现命令模式(转)
    异步回收fork出的子进程(僵尸进程)
  • 原文地址:https://www.cnblogs.com/manmanjiangQwQ/p/13341647.html
Copyright © 2011-2022 走看看