【學習筆記】fwt&&fmt&&子集卷積

FWT

基本思路:將多項式變成點值表達,點值相乘以後再逆變換回來獲得特定形式的卷積;html

多項式的次數界都爲\(2^n\)的形式,\(A_0\)定義爲前一半多項式(下標二進制第一位爲\(0\)),\(A_1\)同理定義;c++

\((A,B)\)表示多項式\(A\)\(B\)的直接拼接,FWT的結果都是一個點值表達,相乘表示點值相乘;優化

下面這些變換都知足線性,記\(n\)爲二進制位數,複雜度:\(O(n\times 2^n)\)spa

or卷積

  • 形式:
    \[ (A|B)_{k} = \sum_{i|j=k}A_i\times B_j \]code

  • 定義變換:
    \[ FWT(A) = (FWT(A_0),FWT(A_0+A_1)) \]htm

    則:\(FWT(A|B)=FWT(A) \times FWT(B)\)blog

  • 概括證實:get

  • 注意到\(A|B=(A_0|B_0,A_0|B_1+A_1|B_0+A_1|B_1)\) ;
    \[ \begin{align} &FWT(A|B)\\ &=FWT(A_0|B_0,A_0|B_1+A_1|B_0+A_1|B_1)\\ &=(FWT(A_0|B_0),FWT(A_0|B_0+A_0|B_1+A_1|B_0+A_1|B_1))\\ &=(FWT(A_0)\times FWT(B_0),FWT(A_0)\times FWT(B_0)+FWT(A_0) \\ &\times FWT(B_1)+FWT(A_1)\times FWT(B_0)+FWT(A_1)\times FWT(B_1))\\ &=(FWT(A_0),FWT(A_0+A_1))\times(FWT(B_0),FWT(B_0+B_1))\\ &=FWT(A)\times FWT(B) \end{align} \]博客

  • 逆變換:
    \[ IFWT(A) = (IFWT(A_0),IFWT(A_1-A_0)) \]it

  • 代碼:

void fwt(int*A,int F){
    for(int i=1;i<n;i<<=1)
    for(int j=0;j<n;j+=i<<1)
    for(int k=0;k<i;++k){
        if(~F)A[j+k+i]=(A[j+k+i]+A[j+k])%mod;
        else A[j+k+i]=(A[j+k+i]-A[j+k]+mod)%mod;
    }
}

and卷積

  • 形式:
    \[ (A\&B)_{k} = \sum_{i\&j=k}A_i\times B_j \]

  • 定義變換:
    \[ FWT(A) = (FWT(A_0+A_1),FWT(A_1)) \]

    則:\(FWT(A\&B)=FWT(A) \times FWT(B)\)

  • 概括證實:

    注意到:\(A\&B=(A_0\&B_0+A_0\&B_1+A_1\&B_0,A_1\&B_1)\)

    同理..

  • 逆變換:
    \[ IFWT(A)=(IFWT(A_0-A_1),IFWT(A_1)) \]

  • 代碼

    void fwt(int*A,int F){
          for(int i=1;i<n;i<<=1)
        for(int j=0;j<n;j+=i<<1)
        for(int k=0;k<i;++k){
            if(~F)A[j+k]=(A[j+k]+A[j+k+i])%mod;
          else A[j+k]=(A[j+k]-A[j+k+i]+mod)%mod;
        }
    }

xor卷積

  • 形式
    \[ (A \oplus B)_{k} = \sum_{i \oplus j=k}A_i\times B_j \]

  • 定義變換:
    \[ FWT(A) = (FWT(A_0+A_1),FWT(A_0-A_1)) \]

    則:\(FWT(A \oplus B)=FWT(A) \times FWT(B)\)

  • 概括證實:

    注意到:\(A \oplus B=(A_0 \oplus B_0+A_1\oplus B_1 ,A_0\oplus B_1+A_1\oplus B_0)\)

    同理..

  • 逆變換:
    \[ IFWT(A)=(IFWT(A_0+A_1)/2,IFWT(A_0-A_1)/2) \]

  • 代碼:

    void fwt(int*A,int F){
          for(int i=1;i<n;i<<=1)
        for(int j=0;j<n;j+=i<<1)
        for(int k=0;k<i;++k){
            int x=A[j+k],y=A[j+k+i];
            A[j+k]=(x+y)%mod,A[j+k+i]=(x-y+mod)%mod;
          if(!~F)A[j+k]=(ll)A[j+k]*iv2%mod,A[j+k+i]=(ll)A[j+k+i]*iv2%mod;
        }
    }

FMT

基本思路:將二進制的每一位當作一維,枚舉每一維作前綴和就能夠獲得集合的前綴和;

直接上代碼了......

子集前綴和(= or)

for(int i=0;i<n;++i)
for(int j=1<<i;j<1<<n;++j)
    if(j>>i&1)f[j]+=f[j^(1<<i)];

超集前綴和(= and)

for(int i=0;i<n;++i)
for(int j=(1<<n)-1;j>=1<<i;--j)
    if(j>>i&1)f[j^(1<<i)]+=f[j];

fmt的逆變換隻須要將第二維反過來減便可;

優勢是代碼短,常數小,複雜度也是\(O(n \times 2^n)\)

子集卷積和

  • \(S,T\)是集合,求\(C:C_S = \sum_{S \subset T} A_T \times B_{S-T}\)  

  • \(A_{i,S}=[|S|=i]\times A_S\)

  • \(C_i = \sum_{j\le i} A_{j}|B_{i-j}\)

  • 這樣就把問題變成了or卷積

  • 因爲咱們只關心知足\(|T|=i\)\(C_{i,T}\),因此是不會多算的;

  • 時間複雜度:\(O(n^2 \times 2^n)\)

  • 【WC2018】州區劃分

    #include<bits/stdc++.h>
    #define ll long long 
    #define mod 998244353
    #define rg register 
    #define il inline 
    using namespace std;
    const int N=22;
    int n,m,p,f[N][1<<N],g[N][1<<N],d[N],fa[N],all,ny[3010];
    int w[1<<N],cnt[1<<N],iw[1<<N],e[N];
    il void inc(int&x,int y){x+=y;if(x>=mod)x-=mod;}
    il void dec(int&x,int y){x-=y;if(x<0)x+=mod;}
    il void fwt(int*A,int F){
      if(~F){
          for(rg int i=0;i<n;++i)
          for(rg int x=1<<i,j=x;j<all;++j)
          if(j&x)inc(A[j],A[j^x]);
      }else{
          for(rg int i=0;i<n;++i)
          for(rg int x=1<<i,j=all-1;~j;--j)
          if(j&x)dec(A[j],A[j^x]);
      }
    }
    int find(int x){return fa[x]==x?x:fa[x]=find(fa[x]);}
    il bool check(int S){
      for(rg int i=0;i<n;++i)fa[i]=i,d[i]=0;
      for(int i=0;i<n;++i)if(S>>i&1)
      for(int j=i+1;j<n;++j)if(S>>j&1){
          if(!(e[i]>>j&1))continue;
          d[i]++,d[j]++;
          int fu=find(i),fv=find(j);
          if(fu!=fv)fa[fu]=fv;
      }
      int lst=-1;
      for(rg int i=0;i<n;++i)if(S>>i&1){
          if(!~lst)lst=find(i);
          if(lst!=find(i) || (d[i]&1))return true;
      }
      return false;
    }
    il int val(int x){
      if(!p)return 1;
      if(p&1)return x;
      return (ll)x*x%mod;
    }
    int main(){
    //    freopen("walk.in","r",stdin);
    //    freopen("walk.out","w",stdout);
      scanf("%d%d%d",&n,&m,&p);all=1<<n;
      for(rg int i=0;i<m;++i){
          int u,v;
          scanf("%d%d",&u,&v),u--,v--;
          e[u]|=1<<v;e[v]|=1<<u;
      }
      for(rg int i=0;i<n;++i)scanf("%d",&w[1<<i]);
      ny[1]=1;for(rg int i=2;i<=2100;++i)ny[i]=(ll)(mod-mod/i)*ny[mod%i]%mod;
      fwt(w,1);
      for(rg int i=0;i<all;++i){  
          iw[i]=ny[w[i]];
          iw[i]=val(iw[i]);
          w[i]=val(w[i]);
          cnt[i]=cnt[i>>1]+(i&1);
      }
      for(rg int i=0;i<all;++i)if(check(i))g[cnt[i]][i]=w[i];
      for(rg int i=0;i<=n;++i)fwt(g[i],1);
      f[0][0]=1;fwt(f[0],1);
      for(rg int i=1;i<=n;++i){
          int *F=f[i];//不加尋址優化是會T的
          for(rg int j=1;j<=i;++j){
              int *a=f[i-j],*b=g[j];
              for(rg int k=0;k<all;++k){
                  inc(F[k],(ll)a[k]*b[k]%mod);
              }
          }
          fwt(F,-1);
          for(rg int j=0;j<all;++j)F[j]=cnt[j]==i?(ll)F[j]*iw[j]%mod:0;
          if(i!=n)fwt(f[i],1);
      }
      cout<<f[n][all-1]<<endl;
      return 0;
    }
相關文章
相關標籤/搜索