1. 程式人生 > >【BZOJ5093】圖的價值(第二類斯特林數,組合數學,NTT)

【BZOJ5093】圖的價值(第二類斯特林數,組合數學,NTT)

ble math n) .cn fin eve 都是 max online

【BZOJ5093】圖的價值(第二類斯特林數,組合數學,NTT)

題面

BZOJ

題解

單獨考慮每一個點的貢獻:
因為不知道它連了幾條邊,所以枚舉一下
\[\sum_{i=0}^{n-1}C_{n-1}^i·i^k·2^{\frac{n(n-1)}{2}}\]
因為有\(n\)個點,所以還要乘以一個\(n\)
所以,我們真正要求的就是:
\[\sum_{i=0}^{n-1}C_{n-1}^i·i^k\]
怎麽做?
看到了\(i^k\)想到了第二類斯特林數
\[m^n=\sum_{i=0}^{m}C_{m}^{i}·S(n,i)·i!\]
所以把這個東西帶回去
\[\sum_{i=0}^{n-1}C_{n-1}^i·i^k\]


\[=\sum_{i=0}^{n-1}C_{n-1}^i·\sum_{j=0}^{i}S(k,j)·C_{i}^{j}·j!\]
如果\(n\)在前面是沒法算的,即使\(O(N)\)也是不行的
所以把後面的\(j\)丟到前面去
\[\sum_{j=0}^{n-1}S(k,j)·j!\sum_{i=j}^{n-1}C_{n-1}^iC_{i}^j\]
後面那個是啥呢?
我們來考慮一下組合意義
\(n-1\)個球從中選出\(i\)個染成黑色
再從\(i\)個黑球中選出\(j\)個染成白色
既然染成白色的球固定是\(j\)
那麽,我可以想先從\(n-1\)個球中選出\(j\)個直接染成白色
因為\(i\)
個枚舉的,相當於我可以取出任意個數染成黑色
既然有\(j\)個白球了,剩下\(n-1-j\)個球,染色或者不染色都是可以的
所以就再乘上\(2^{n-1-j}\)

\[\sum_{j=0}^{n-1}S(k,j)·j!\sum_{i=j}^{n-1}C_{n-1}^iC_{i}^j\]
\[=\sum_{j=0}^{n-1}S(k,j)·j!·C_{n-1}^j·2^{n-1-j}\]
\[=\sum_{j=0}^{n-1}S(k,j)·j!·\frac{(n-1)!}{j!(n-j-1)!}·2^{n-1-j}\]
\[=\sum_{j=0}^{n-1}S(k,j)·j!·\frac{(n-1)!}{j!(n-j-1)!}·2^{n-1-j}\]


\[=\sum_{j=0}^{n-1}S(k,j)·\frac{(n-1)!}{(n-j-1)!}·2^{n-1-j}\]
至於\(S(k,j)\)怎麽算?
不要忘記第二類斯特林數也是一個卷積的形式
戳這裏去看看
那麽,先算出第二類斯特林數,直接算就好啦
當然啦,對於\(j>k\),\(S(k,j)=0\)就不用枚舉了
所以最多枚舉到\(k\)
復雜度\(O(klogk)\)

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<set>
#include<map>
#include<vector>
#include<queue>
using namespace std;
#define ll long long
#define RG register
#define MOD 998244353
#define MAX 1000000
const int pr=3;
const int phi=MOD-1;
int fpow(int a,int b)
{
    int s=1;
    while(b){if(b&1)s=1ll*s*a%MOD;a=1ll*a*a%MOD;b>>=1;}
    return s;
}
int N,M,l,a[MAX],b[MAX],S[MAX],r[MAX];
void NTT(int *P,int opt)
{
    for(int i=0;i<N;++i)if(i<r[i])swap(P[i],P[r[i]]);
    for(int i=1;i<N;i<<=1)
    {
        int W=fpow(pr,phi/(i<<1));
        for(int p=i<<1,j=0;j<N;j+=p)
        {
            int w=1;
            for(int k=0;k<i;++k,w=1ll*w*W%MOD)
            {
                int X=P[j+k],Y=1ll*w*P[i+j+k]%MOD;
                P[j+k]=(X+Y)%MOD;P[i+j+k]=((X-Y)%MOD+MOD)%MOD;
            }
        }
    }
    if(opt==-1)reverse(&P[1],&P[N]);
}
void Work()
{
    M+=N;
    for(N=1;N<=M;N<<=1)++l;
    for(int i=0;i<N;++i)r[i]=(r[i>>1]>>1)|((i&1)<<(l-1));
    NTT(a,1);NTT(b,1);
    for(int i=0;i<N;++i)a[i]=1ll*a[i]*b[i]%MOD;
    NTT(a,-1);
    for(int i=0,inv=fpow(N,MOD-2);i<N;++i)a[i]=1ll*a[i]*inv%MOD;
}
int n,K,jc[MAX],inv[MAX],ans;
int main()
{
    scanf("%d%d",&n,&K);
    jc[0]=inv[0]=1;
    for(int i=1;i<=K;++i)jc[i]=1ll*jc[i-1]*i%MOD;
    for(int i=1;i<=K;++i)inv[i]=fpow(jc[i],MOD-2);
    N=M=K;
    for(int i=0;i<=K;++i)a[i]=(i&1)?MOD-inv[i]:inv[i];
    for(int i=0;i<=K;++i)b[i]=1ll*fpow(i,K)*inv[i]%MOD;
    Work();
    for(int i=0;i<=K;++i)S[i]=a[i];
    int inv2=fpow(2,MOD-2);
    for(int i=0,p=fpow(2,n-1),pp=1;i<=min(n-1,K);++i)
    {
        int t=1ll*S[i]*pp%MOD*p%MOD;
        p=1ll*p*inv2%MOD;
        pp=1ll*pp*(n-1-i)%MOD;
        ans=(ans+t)%MOD;
    }
    ans=1ll*ans*n%MOD;
    ans=1ll*ans*fpow(2,1ll*(n-1)*(n-2)/2%phi)%MOD;
    printf("%d\n",ans);
    return 0;
}

【BZOJ5093】圖的價值(第二類斯特林數,組合數學,NTT)