• [学习笔记]分治FFT


    一般的分治FFT是指:

    https://www.luogu.org/problemnew/show/P4721

    考虑后面的f和前面的f有关系,但是贡献可以分着计算,逐一累计上去。

    考虑cdq分治。算出前面的[1,mid]的f之后,可以直接一次NTT,把后面[mid+1,r]的f的一部分算出来,累加上去。

    对于后面的部分,发现都是一个前缀没有计算上。继续分治下去即可。

    画个图就是这样。

    细节注意:

    1.边界,

    2.0~n-1

    3.四倍N的数组

    4.注意之后每次都是NTT一个前缀。

    #include<bits/stdc++.h>
    #define reg register int
    #define il inline
    #define numb (ch^'0')
    #define int long long
    using namespace std;
    typedef long long ll;
    il void rd(int &x){
        char ch;x=0;bool fl=false;
        while(!isdigit(ch=getchar()))(ch=='-')&&(fl=true);
        for(x=numb;isdigit(ch=getchar());x=x*10+numb);
        (fl==true)&&(x=-x);
    }
    namespace Miracle{
    const int N=1e5+5;
    const int mod=998244353;
    const int G=3;
    const int GI=332748118;
    int n,m;
    ll qm(ll x,ll y){
        ll ret=1;
        while(y){
            if(y&1) ret=ret*x%mod;
            x=x*x%mod;
            y>>=1;
        }
        return ret;
    }
    int rev[4*N];
    void fft(int *a,int n,int c){
        for(reg i=0;i<=n-1;++i){
            if(i<rev[i]) swap(a[i],a[rev[i]]);
        }
        for(reg p=2;p<=n;p<<=1){
            ll gen;
            if(c==1) gen=qm(G,(mod-1)/p);
            else gen=qm(GI,(mod-1)/p); 
            for(reg l=0;l<n;l+=p){
                ll lp=1;
                for(reg k=l;k<l+p/2;++k){
                    ll tmp=a[k+p/2];
                    a[k+p/2]=(a[k]-tmp*lp%mod+mod)%mod;
                    a[k]=(a[k]+tmp*lp%mod)%mod;
                    lp=lp*gen%mod;
                }
            }
        }
    }
    
    ll g[4*N],f[4*N],c[4*N],d[4*N];
    void calc(int *a,int *b,int n){
        for(reg i=0;i<n;++i){
            rev[i]=(rev[i>>1]>>1)|((i&1)?n>>1:0);
        }
        fft(a,n,1);fft(b,n,1);
        for(reg i=0;i<n;++i) b[i]=a[i]*b[i]%mod;
        fft(b,n,-1);
        ll inv=qm(n,mod-2);
        for(reg i=0;i<n;++i) b[i]=b[i]*inv%mod;
    }
    void divi(int l,int r,int L,int R){
        //cout<<" divi "<<l<<" "<<r<<" and "<<L<<" "<<R<<endl;
        if(l==r){
            return;
        }
        int mid=(l+r)>>1;
        int Md=(L+R)>>1;
        divi(l,mid,L,Md);
        
        //cout<<" bac to "<<l<<" "<<r<<endl;
        for(reg i=l;i<=mid;++i) c[i-l]=f[i];
        for(reg i=mid+1;i<=r;++i) c[i-l]=0;
        for(reg i=L;i<=R;++i) d[i-L]=g[i];
        for(reg i=r-l+1;i<=(r-l+1)*2-1;++i) c[i]=d[i]=0;
        calc(c,d,(r-l+1)*2);
        for(reg i=mid+1;i<=r;++i) f[i]=(f[i]+d[i-l])%mod;
        //cout<<" f[4] "<<f[4]<<" f[5] "<<f[5]<<endl;
        
        divi(mid+1,r,L,Md);
    }
    int main(){
        rd(n);
        int lp=n;
        for(reg i=1;i<n;++i) rd(g[i]);
        g[0]=0;f[0]=1;
        for(m=n,n=1;n<m;n<<=1);
        divi(0,n-1,0,n-1);
        for(reg i=0;i<lp;++i){
            printf("%lld ",f[i]);
        }
        return 0;
    }
    
    }
    signed main(){
        Miracle::main();
        return 0;
    }
    
    /*
       Author: *Miracle*
       Date: 2018/12/21 14:08:16
    */

     配赠福利:

    一、

    升级版:真的分治fft(这个代码我觉得如果有l>=r-l+1,那么可以直接return掉,后面[l,mid]*[l,mid]没有意义。)

    现在的g变成了f,直接刚才那样cdq,会出现一些mid+1~r区间的f还要贡献,但是我们目前没有计算出来

    还是考虑cdq分治

    假设计算出来了[l,mid],那么,先把[l,mid]*[l,mid]的多项式的贡献计算出来

    剩下没有算出来的怎么补?

    每个值剩下的没有计算的部分,其右部分没有被计算到的区间,一定是一个l>=r-l+1的区间

    如果有l>=r-l+1,那么把f[0,r-l]*f[l,mid]再计算一下,然后*2(其实本质上是补全第一次乘漏的部分)

    是一种延迟处理的方法,因为先算的话,有一半没有计算出来;而反过来再算的时候,涉及到的f就已经都算完了。恰好,两边对称,所以*2解决。

    二、

    分治FFT字面意思理解一下的话,,就是分治+FFT。。。

    所以,如果要计算:

    (x+a)*(x+b)*(x+c)*(x+d)*....

    直接暴力算的复杂度是(2+3+4+...n)*logn

    分治的话,每个(x+a)贡献的是2的长度,,一共贡献logn次,所以O(nlog^2n)

  • 相关阅读:
    可以多个广告浮动的页面
    获取内容并截取长度用...替换
    用UIManager使Swing界面转换风格。
    Java FileInputStream
    win7 部署tomcat总结
    观察者模式
    Java RandomAccessFile
    Java 客户端界面功能:停止当前操作
    多线程的开启与管理
    log4j内容记录
  • 原文地址:https://www.cnblogs.com/Miracevin/p/10158596.html
Copyright © 2020-2023  润新知