【BZOJ5093】图的价值(第二类斯特林数,组合数学,NTT)

【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;
}

原文地址:https://www.cnblogs.com/cjyyb/p/8456398.html

时间: 2024-07-30 22:25:11

【BZOJ5093】图的价值(第二类斯特林数,组合数学,NTT)的相关文章

bzoj 5093 图的价值 —— 第二类斯特林数+NTT

题目:https://www.lydsy.com/JudgeOnline/problem.php?id=5093 每个点都是等价的,从点的贡献来看,得到式子: \( ans = n * \sum\limits_{d=0}^{n-1} d^{k} * 2^{C_{n-1}^{2}} * C_{n-1}^{d} \) 使用 \( n^{k} = \sum\limits_{i=0}^{k} S(k,i) * i! *C_{n}^{i} \) 得到 \( ans = n * \sum\limits_{d

P4091 [HEOI2016/TJOI2016]求和(第二类斯特林数,ntt)

题面:https://www.luogu.org/problem/P4091 题解:\[\begin{array}{l}f(n) = \sum\limits_{i = 0}^n {\sum\limits_{j = 0}^i {{\rm{S}}(i,j) \cdot {2^{\rm{j}}} \cdot j!} } \\ = \sum\limits_{i = 0}^n {\sum\limits_{j = 0}^n {{\rm{S}}(i,j) \cdot {2^{\rm{j}}} \cdot j!

【bzoj5093】[Lydsy1711月赛]图的价值(NTT+第二类斯特林数)

题意: 给定\(n\)个点,一个图的价值定义为所有点的度数的\(k\)次方之和. 现在计算所有\(n\)个点的简单无向图的价值之和. 思路: 将式子列出来: \[ \sum_{i=1}^n\sum_{j=0}^{n-1}{n-1\choose j}2^{\frac{(n-1)(n-2)}{2}}j^k \] 表示分别考虑每个点的贡献,我们只需要枚举其度数即可,其余的边任意连. 然后我们将后面的\(j^k\)用第二类斯特林数展开: \[ \begin{aligned} &\sum_{i=1}^{n

bzoj5093:图的价值(第二类斯特林数+NTT)

传送门 首先,题目所求为\[n\times 2^{C_{n-1}^2}\sum_{i=0}^{n-1}C_{n-1}^ii^k\] 即对于每个点\(i\),枚举它的度数,然后计算方案.因为有\(n\)个点,且关于某个点连边的时候剩下的边都可以随便连,所以有前面的两个常数 所以真正要计算的是\[\sum_{i=0}^{n-1}C_{n-1}^ii^k\] 根据第二类斯特林数的性质,有\[i^k=\sum_{j=0}^iS(k,j)\times j!\times C_i^j\] 然后带入,得\[\s

bzoj 5093 [Lydsy1711月赛]图的价值 NTT+第二类斯特林数

[Lydsy1711月赛]图的价值 Time Limit: 30 Sec  Memory Limit: 256 MBSubmit: 245  Solved: 128[Submit][Status][Discuss] Description “简单无向图”是指无重边.无自环的无向图(不一定连通). 一个带标号的图的价值定义为每个点度数的k次方的和. 给定n和k,请计算所有n个点的带标号的简单无向图的价值之和. 因为答案很大,请对998244353取模输出. Input 第一行包含两个正整数n,k(

【cf961G】G. Partitions(组合意义+第二类斯特林数)

传送门 题意: 给出\(n\)个元素,每个元素有价值\(w_i\).现在要对这\(n\)个元素进行划分,共划分为\(k\)组.每一组的价值为\(|S|\sum_{i=0}^{|S|}w_i\). 最后询问所有划分的总价值. 思路: 直接枚举划分不好计算,考虑单独计算每一个元素的贡献,那么就有式子: \[ \sum_{i=1}^nw_i\sum_{j=1}^{n-k+1}{n-1\choose j-1}\begin{Bmatrix} n - j \\ k - 1 \end{Bmatrix}j \]

Gym 101147G 第二类斯特林数

大致题意: n个孩子,k场比赛,每个孩子至少参加一场比赛,且每场比赛只能由一个孩子参加.问有多少种分配方式. 分析: k>n,就无法分配了. k<=n.把n分成k堆的方案数乘以n的阶乘.N分成k堆得方案数即第二类斯特林数 http://blog.csdn.net/acdreamers/article/details/8521134 #include <bits/stdc++.h> using namespace std; typedef long long ll; const ll

Light OJ 1236 Race 第二类斯特林数

第二类斯特林数 n 匹马 分成1 2 3... n组 每一组就是相同排名 没有先后 然后组与组之间是有顺序的 在乘以组数的阶乘 #include <cstdio> #include <cstring> using namespace std; int dp[1010][1010]; int a[1010]; int main() { a[0] = 1; dp[0][0] = 1; for(int i = 1; i <= 1000; i++) { dp[i][0] = 0; d

swjtu oj Paint Box 第二类斯特林数

http://swjtuoj.cn/problem/2382/ 题目的难点在于,用k种颜色,去染n个盒子,并且一定要用完这k种颜色,并且相邻的格子不能有相同的颜色, 打了个表发现,这个数是s(n, k) * k! s(n, k)表示求第二类斯特林数. 那么关键是怎么快速求第二类斯特林数. 这里提供一种O(k)的算法. 第二类斯特林数: #include <cstdio> #include <cstdlib> #include <cstring> #include <