[BZOJ5093]图的价值(NTT+第二类Stirling数)
5093: [Lydsy1711月赛]图的价值
Time Limit: 30 Sec Memory Limit: 256 MB
Submit: 250 Solved: 130
[Submit][Status][Discuss]Description
“简单无向图”是指无重边、无自环的无向图(不一定连通)。一个带标号的图的价值定义为每个点度数的k次方的和。给定n和k,请计算所有n个点的带标号的简单无向图的价值之和。因为答案很大,请对998244353取模输出。Input
第一行包含两个正整数n,k(1<=n<=10^9,1<=k<=200000)。Output
输出一行一个整数,即答案对998244353取模的结果。
Sample Input
6 5Sample Output
67584000HINT
Source
挺喜欢的一道题,首先从每个点的贡献的角度考虑并列出式子,然后通过Stirling数将幂数转化为下降幂,再交换求和枚举顺序并用组合数公式化简,最后用Stirling数组合意义公式转化成可以FFT的形式,直接套用NTT即可。
具体参考https://blog.csdn.net/cdsszjj/article/details/79080229
首先第一步列出式子不难想到,然后观察发现这是一个n项求和,显然不能直接做,第二步引入Stirling数,尽量使关于n的项消去。通过把与n有关的项放在等式最后并变形,成功使枚举上界从n变为min(n,k),大大降低了时间复杂度。但是递推求$O(k)$的Stirling数仍然无法承受,但第二类Stirling数有恰好符合FFT形式的公式,我们另$a[i]=\frac{(-1)^i}{i!}$,$b[i]=\frac{(m-i)^n}{(m-i)!}$,结果就是最基本的FFT形式:$S[n]=\sum_{i=0}^{n}a[i]b[n-i]$。这里将NTT封装起来感觉清晰了不少。
扯几句题外话:
1.设$f(n)=\sum\limits_{i=1}^n(i,n)$,$g(n)=\sum\limits_{d|n}f(d)$,化简$g(n)$。
$$f(n)=\sum\limits_{i=1}^n\sum\limits_{d|i\ d|n}\phi(d)=\sum\limits_{d|n}\phi(d)\frac{n}{d}$$到这里我们可以不断带入化简,但是我们其实可以发现:$$f(n)=\phi*id$$$$g(n)=I*f=I*(\phi*id)=(I*\phi)*id=id*id$$所以$$g(n)=\sum\limits_{d|n}d\frac{n}{d}$$
2.几个组合数化简:$$\sum_{i=0}^nC_n^iC_i^k=C_n^k\sum_{i=k}^nC_{n-k}^{i-k}$$$$\sum_{i=0}^aC_a^iC_b^{k-i}=C_{a+b}^k$$$$\sum_{i=0}^aC_a^iC_a^i=C_{2a}^i$$$$\sum_{i=0}^aC_a^{a-i}C_b^{i+k}=C_{a+b}^{a+k}$$
#include<cstdio>
#include<algorithm>
#define rep(i,l,r) for (int i=l; i<=r; i++)
typedef long long ll;
using namespace std; const int N=,mod=,G=;
int n,k,ans,a[N],s[N],fac[N],fin[N],fn[N]; int ksm(int a,ll b){
int res;
for (res=; b; a=(1ll*a*a)%mod,b>>=)
if (b & ) res=(1ll*res*a)%mod;
return res;
} namespace NTT{
int n,L,rev[N];
void init(int m){
n=; L=;
while (n<=m) n<<=,L++;
for (int i=; i<n; i++) rev[i]=(rev[i>>]>>)|((i&)<<(L-));
}
void DFT(int a[],int f){
for (int i=; i<n; i++) if (i<rev[i]) swap(a[i],a[rev[i]]);
for (int i=; i<n; i<<=){
int wn=ksm(G,(f==)?(mod-)/(i<<):(mod-)-(mod-)/(i<<));
for (int p=i<<,j=; j<n; j+=p){
int w=;
for (int k=; k<i; k++,w=1ll*w*wn%mod){
int x=a[j+k],y=1ll*w*a[i+j+k]%mod; a[j+k]=(x+y)%mod; a[i+j+k]=(x-y+mod)%mod;
}
}
}
if (f==) return;
int inv=ksm(n,mod-);
for (int i=; i<n; i++) a[i]=1ll*a[i]*inv%mod;
}
void multi(int a[],int b[]){
DFT(a,); DFT(b,);
for (int i=; i<n; i++) a[i]=(1ll*a[i]*b[i])%mod;
DFT(a,-);
}
} int main(){
freopen("bzoj5093.in","r",stdin);
freopen("bzoj5093.out","w",stdout);
scanf("%d%d",&n,&k); n--; fac[]=fin[]=fn[]=;
rep(i,,k) fac[i]=1ll*fac[i-]*i%mod;
fin[k]=ksm(fac[k],mod-);
for (int i=k-; i; i--) fin[i]=(1ll*fin[i+]*(i+))%mod;
rep(i,,k) fn[i]=(1ll*fn[i-]*(n-i+))%mod;
rep(i,,k) s[i]=(((i&)?-:)*fin[i]+mod)%mod;
rep(i,,k) a[i]=(1ll*ksm(i,k)*fin[i])%mod;
NTT::init(*k+); NTT::multi(s,a);
rep(i,,min(n,k)) ans=(ans+1ll*s[i]*fac[i]%mod*fn[i]%mod*fin[i]%mod*ksm(,n-i))%mod;
printf("%lld\n",1ll*ans*(n+)%mod*ksm(,1ll*n*(n-)/)%mod);
return ;
}
[BZOJ5093]图的价值(NTT+第二类Stirling数)的更多相关文章
- bzoj 5093 [Lydsy1711月赛]图的价值 NTT+第二类斯特林数
[Lydsy1711月赛]图的价值 Time Limit: 30 Sec Memory Limit: 256 MBSubmit: 245 Solved: 128[Submit][Status][D ...
- bzoj5093图的价值:多项式,斯特林数(二项式反演)
Description “简单无向图”是指无重边.无自环的无向图(不一定连通). 一个带标号的图的价值定义为每个点度数的k次方的和. 给定n和k,请计算所有n个点的带标号的简单无向图的价值之和. 因为 ...
- bzoj5093:图的价值(第二类斯特林数+NTT)
传送门 首先,题目所求为\[n\times 2^{C_{n-1}^2}\sum_{i=0}^{n-1}C_{n-1}^ii^k\] 即对于每个点\(i\),枚举它的度数,然后计算方案.因为有\(n\) ...
- 【BZOJ5093】图的价值(第二类斯特林数,组合数学,NTT)
[BZOJ5093]图的价值(第二类斯特林数,组合数学,NTT) 题面 BZOJ 题解 单独考虑每一个点的贡献: 因为不知道它连了几条边,所以枚举一下 \[\sum_{i=0}^{n-1}C_{n-1 ...
- 【题解】BZOJ5093图的价值(二项式+NTT)
[题解]BZOJ5093图的价值(二项式+NTT) 今天才做这道题,是我太弱了 强烈吐槽c++这种垃圾语言tmd数组越界不re反倒去别的数组里搞事情我只想说QAQ 推了一张A4纸的式子 考虑每个点的度 ...
- [CF932E]Team Work & [BZOJ5093]图的价值
CF题面 题意:求\(\sum_{i=0}^{n}\binom{n}{i}i^k\) \(n\le10^9,k\le5000\) 模\(10^9+7\) BZOJ题面 题意:求\(n*2^{\frac ...
- lightOJ 1326 Race(第二类Stirling数)
题目链接:http://lightoj.com/volume_showproblem.php?problem=1326 题意:有n匹马赛跑.问有多少种不同的排名结果.可以有多匹马的排名相同. 思路:排 ...
- [总结] 第二类Stirling数
上一道例题 我们来介绍第二类Stirling数 定义 第二类Stirling数实际上是集合的一个拆分,表示将n个不同的元素拆分成m个集合的方案数,记为 或者 .和第一类Stirling数不同的是,集合 ...
- 第二类Stirling数
第二类斯特林数 第二类Stirling数:S2(p, k) 1.组合意义:第二类Stirling数计数的是把p个互异元素划分为k个非空集合的方法数 2.递推公式: S2(0, 0) = 1 S2(p, ...
随机推荐
- 【BZOJ】4293: [PA2015]Siano 线段树上二分
[题意]给定n棵高度初始为0的草,每天每棵草会长高a[i],m次收割,每次在d[i]天将所有>b[i]的草收割到b[i],求每次收割量.n<=500000. [算法]线段树上二分 [题解] ...
- 解决Chrome下表单自动填充后背景色为黄色
Chrome浏览器在表单自动填充后会显示黄色背景,这是Chrome的私有属性导致,对于有洁癖的人来讲,是不喜欢的,我们可以手动去掉. 代码如下: input:-webkit-autofill { -w ...
- Master of Phi (欧拉函数 + 积性函数的性质 + 狄利克雷卷积)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6265 题目大意:首先T是测试组数,n代表当前这个数的因子的种类,然后接下来的p和q,代表当前这个数的因 ...
- [Gym-100625J] 搜索
题目链接:https://cn.vjudge.net/problem/Gym-100625J 具体思路:首先,具体思路是两个人一起走到一个点,然后两个人按照同样的道路走出去,听了别人的思路,还有一种特 ...
- 如何通过掩码计算可用的IP数量
假设掩码是28,28也就是28个1.本身掩码是255.255.255.255那么转换成二进制也就是 11111111,11111111,11111111,11111111 那么28个1也就是: 111 ...
- Freemaker 自定义指令和函数
自定义函数和指令都可以在前台或者后台进行指定. 个人理解:指令的作用,主要是进行页面调整之后进行输出:函数的作用,主要是为了进行运算,返回运算结果供前台展示. (一) 自定义指令 使用以下格式调用自定 ...
- CPU运行时间——time
用途说明time命令常用于测量一个命令的运行时间,注意不是用来显示和修改系统时间的(这是date命令干的事情).但是今天我通过查看time命令的手册页,发现它能做的不仅仅是测量运行时间,还可以测量内存 ...
- 用WP SMTP插件实现邮件发送功能
WordPress本身是采用mail()函数发邮件的,但是这样发出的邮件很容易被放入垃圾箱,很多主机商(特别是Windows主机)为了避免用户滥发邮件直接禁用了mail()函数,还有些云计算平台(比如 ...
- 解决类似 /usr/lib64/libstdc++.so.6: version `GLIBCXX_3.4.21' not found 的问题
https://itbilu.com/linux/management/NymXRUieg.html
- 微软企业库5.0 学习之路——扩展学习篇、库中的依赖关系注入(重构 Microsoft Enterprise Library)[转]
这篇文章是我在patterns & practices看到的一篇有关EntLib5.0的文章,主要介绍了EntLib5.0的这次的架构变化由来,觉得很不错,大家可以看一下! 在过去几年中,依赖 ...