其他分享
首页 > 其他分享> > 【JZOJ100209】【20190705】狂妄之人

【JZOJ100209】【20190705】狂妄之人

作者:互联网

题目

\(S\)串长为\(n\),字符集大小为\(k\)

一次操作为:取走\(S\)的任意一个字符或将\(S\)重排为一个没有出现过的字符\(S'\)

询问有多少个\(S\)使得后手必胜,答案对\(P\)取模

$n \le 3 \times 10^5  , k \le 10^9  ,  10^8 \le P \le 10^9+100 $

题解

Code

#include<bits/stdc++.h>
#define ll long long 
#define il inline 
#define rg register 
using namespace std;
const int N=1<<19,M=20;
int n,m,mx,P,L,a[M][N],cnt[N],b[N],c[N],ny[N],fac[N],inv[N];ll lim;
il void inc(int&x,int y){x+=y;if(x>=P)x-=P;}
il void dec(int&x,int y){x-=y;if(x<0)x+=P;}
il void fwt(int*A){
    for(rg int i=1;i<L;i<<=1)
    for(rg int j=0;j<L;j+=i<<1)
    for(rg int k=0;k<i;++k){
        inc(A[j+k+i],A[j+k]);
    }
}
il void ifwt(int*A){
    for(rg int i=1;i<L;i<<=1)
    for(rg int j=0;j<L;j+=i<<1)
    for(rg int k=0;k<i;++k){
        dec(A[j+k+i],A[j+k]);
    }
}
il void exp(int*A,int*B,int l){
    B[0]=1;
    for(rg int i=1;i<=l;++i){
        B[i]=0;ll t=0;
        for(rg int j=1;j<=i;++j){
            //inc(B[i],1ll*j*A[j]%P*B[i-j]%P);
            t+=1ll*A[j]*B[i-j]%lim*j;
            if(t>=lim)t-=lim;
        }
        //B[i]=1ll*B[i]*ny[i]%P;
        B[i]=(t%P)*ny[i]%P;
    }
}
il void ln(int*A,int*B,int l){
    B[0]=0;
    for(int i=1;i<=l;++i){
        B[i]=0;ll t=0;
        for(rg int j=1;j<i;++j){
            //dec(B[i],1ll*j*B[j]%P*A[i-j]%P);
            t+=1ll*B[j]*A[i-j]%lim*j;
            if(t>=lim)t-=lim;
        }
        //B[i]=(A[i]+1ll*B[i]*ny[i])%P;
        B[i]=(A[i]+(P-t%P)*ny[i])%P;
    }
}
il void pow(int*A,int k,int l){
    static int t1[N];   
    ln(A,t1,l);
    for(int i=0;i<=l;++i)t1[i]=1ll*t1[i]*k%P;
    exp(t1,A,l);
}
int main(){
    freopen("megalovania.in","r",stdin);
    freopen("megalovania.out","w",stdout);
    scanf("%d%d%d",&n,&m,&P);
    if(n&1)return puts("0"),0;
    lim=(ll)4e18/P*P;
    L=1;while(L<=n)L<<=1;
    ny[1]=1;for(int i=2;i<=n;++i)ny[i]=1ll*(P-P/i)*ny[P%i]%P;
    for(int i=fac[0]=inv[0]=1;i<=n;++i){
        fac[i]=1ll*fac[i-1]*i%P;
        inv[i]=1ll*inv[i-1]*ny[i]%P;
        cnt[i]=cnt[i>>1]+(i&1);
    }
    mx=cnt[n];
    for(int i=0;i<=n;++i)if((i|n)==n)a[cnt[i]][i]=inv[i];
    for(int i=0;i<=mx;++i)fwt(a[i]);
    for(int i=0;i<=n;++i)if((i|n)==n){
        for(int j=0;j<=mx;++j)b[j]=a[j][i];
        pow(b,m,mx);
        c[i]=b[mx];
    }
    ifwt(c);
    int ans=1ll*c[n]*fac[n]%P;
    cout<<ans<<endl;
    return 0;
}

标签:狂妄,字符,20190705,frac,奇数,ln,JZOJ100209,int,exp
来源: https://www.cnblogs.com/Paul-Guderian/p/11148048.html