[HEOI2016]求和 sum
[HEOI2016]求和 sum
标签: NTT cdq分治 多项式求逆 第二类斯特林数
Description
求$$\sum_{i=0}n\sum_{j=0}i S(i,j)×2^j×(j!)$$
其中S(i,j)代表第二类斯特林数。
Solution
解法一
记Bell数\(B(n)=\sum_{i=0}^nS(n,i)\)
根据第二类斯特林数的组合意义,\(B(i)\)代表把n个球放进任意个相同的盒子的方案数。
那么有$$B(n)=\sum_{i=0}^{n-1} C(n-1,i) B(i)$$
讨论一下有多少个球与第一个球同盒即可。
记\(B'(n)=\sum_{i=0}^nS(n,i)×i!\)
这个的组合意义是把n个求放进任意个不同盒子的方案数。
有$$B'(n)=\sum_{i=0}^{n-1} C(n,i)B'(i)$$
讨论一下第一个盒子有多少个球
现在记\(f(n)=\sum_{i=0}^nS(n,i)×2^i×i!\)
这个的组合意义是把n个球放进任意个不同且有两种颜色盒子的方案数。
显然$$f(n)=2\sum_{i=0}^{n-1} C(n,i)f(i)$$
相当于每个盒子有两种不同的颜色选。
化简式子
\]
设\(g(i)={2 \over i!}x^i\),特别地,\(g(0)=0\).
这就是一个卷积的形式了。
然后cdq+NTT即可。
解法2
以上面的式子为基础做多项式求逆。
构造一个生成函数\(F(x)=\sum_{i=0}^{\infty}{f(i) \over i!}x^i\)与\(G(x)=\sum _{i=1}^{\infty}g(i)\)
那么根据上面的递推式子,有\(F(x)=F(x)×G(x)+1\)
解得\(F(x)={1 \over {1-G(x)}}\)
解法3
考虑斯特林数的容斥求法。
我们令\(f(i)=\sum_{j=i}^nS(j,i)\)
那么答案显然是\(\sum_{i=0}^nf(i)2^ii!\)
接下来考虑怎么求\(f(i)\)
\\ =\sum_ {j=0}^n {\frac 1 {i!}}\sum_{k=0}^i (-1)^kC(i,k)(i-k)^j
\\ =\sum_ {k=0} {(-1)^k \over k!} {\sum_{j=0}^n (i-k)^j \over (i-k)! }
\]
一遍NTT即可。
Code
解法一
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<cmath>
#include<queue>
#include<stack>
#include<set>
#include<map>
using namespace std;
#define ll long long
#define REP(i,a,b) for(int i=(a),_end_=(b);i<=_end_;i++)
#define DREP(i,a,b) for(int i=(a),_end_=(b);i>=_end_;i--)
#define EREP(i,a) for(int i=start[(a)];i;i=e[i].next)
inline int read()
{
int sum=0,p=1;char ch=getchar();
while(!(('0'<=ch && ch<='9') || ch=='-'))ch=getchar();
if(ch=='-')p=-1,ch=getchar();
while('0'<=ch && ch<='9')sum=sum*10+ch-48,ch=getchar();
return sum*p;
}
const int maxn=4e5+20;
const int mod=998244353;
inline int power(int a,int b)
{
int ans=1;
while(b)
{
if(b & 1)ans=(ll)ans*a%mod;
b>>=1;
a=(ll)a*a%mod;
}
return ans;
}
int n,f[maxn],g[maxn],jc[maxn],jcn[maxn],inv[maxn],rev[maxn];
void init()
{
n=read();
jc[0]=jcn[0]=jc[1]=jcn[1]=inv[1]=1;
REP(i,2,n)inv[i]=(ll)(mod-mod/i)*inv[mod%i]%mod,jc[i]=(ll)i*jc[i-1]%mod,jcn[i]=(ll)inv[i]*jcn[i-1]%mod;
REP(i,1,n)g[i]=jcn[i]*2%mod;
}
int A[maxn],B[maxn];
inline void NTT(int *p,int N,int op)
{
int n=1,l=0;while(n<N)n<<=1,l++;
REP(i,1,n-1)rev[i]=(rev[i>>1]>>1)|((i&1)<<(l-1));
REP(i,0,n-1)if(i<rev[i])swap(p[i],p[rev[i]]);
for(int i=1;i<n;i<<=1)
{
int W=power(3,(mod-1)/(i<<1));
for(int j=0;j<n;j+=i<<1)
{
int w=1;
for(int k=j;k<i+j;k++,w=(ll)w*W%mod)
{
int x=p[k],y=(ll)p[k+i]*w%mod;
p[k]=x+y;p[k+i]=x-y;
if(p[k]>=mod)p[k]-=mod;
if(p[k+i]<0)p[k+i]+=mod;
}
}
}
if(op==-1)
{
int inv=power(n,mod-2);
REP(i,0,n-1)p[i]=(ll)p[i]*inv%mod;
reverse(p+1,p+n);
}
}
void solve(int l,int r)
{
if(l==r)return;
int mid=(l+r)>>1;
solve(l,mid);
REP(i,0,mid-l)A[i]=f[i+l];
REP(i,0,r-l)B[i]=g[i];
int N=1;
while(N<=(mid-l)+(r-l)+1)N<<=1;
NTT(A,N,1);NTT(B,N,1);
REP(i,0,N-1)A[i]=(ll)A[i]*B[i]%mod;
NTT(A,N,-1);
REP(i,mid-l+1,r-l)f[i+l]=(A[i]+f[i+l])%mod;
REP(i,0,N-1)A[i]=B[i]=0;
solve(mid+1,r);
}
void doing()
{
f[0]=1;
solve(0,n);
int ans=0;
REP(i,0,n)ans=(ans+(ll)f[i]*jc[i])%mod;
printf("%d\n",ans);
}
int main()
{
init();
doing();
return 0;
}
解法二
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<cmath>
#include<queue>
#include<stack>
#include<set>
#include<map>
using namespace std;
#define ll long long
#define REP(i,a,b) for(int i=(a),_end_=(b);i<=_end_;i++)
#define DREP(i,a,b) for(int i=(a),_end_=(b);i>=_end_;i--)
#define EREP(i,a) for(int i=start[(a)];i;i=e[i].next)
inline int read()
{
int sum=0,p=1;char ch=getchar();
while(!(('0'<=ch && ch<='9') || ch=='-'))ch=getchar();
if(ch=='-')p=-1,ch=getchar();
while('0'<=ch && ch<='9')sum=sum*10+ch-48,ch=getchar();
return sum*p;
}
const int maxn=3e5+20;
const int mod=998244353;
int n,jc[maxn],jcn[maxn],inv[maxn],rev[maxn];
inline int power(int a,int b)
{
int ans=1;
while(b)
{
if(b & 1)ans=(ll)ans*a%mod;
b>>=1;
a=(ll)a*a%mod;
}
return ans;
}
inline void init()
{
n=read();jc[0]=jcn[0]=jc[1]=jcn[1]=inv[1]=1;
REP(i,2,n)inv[i]=(ll)(mod-mod/i)*inv[mod%i]%mod,jc[i]=(ll)i*jc[i-1]%mod,jcn[i]=(ll)inv[i]*jcn[i-1]%mod;
}
int g[maxn],ans[maxn],A[maxn],B[maxn],C[maxn];
inline void NTT(int *p,int N,int op)
{
int n=1,l=0;while(n<N)n<<=1,l++;
REP(i,0,n-1)rev[i]=(rev[i>>1]>>1)|((i&1)<<(l-1));
REP(i,0,n-1)if(i<rev[i])swap(p[i],p[rev[i]]);
for(int i=1;i<n;i<<=1)
{
int W=power(3,(mod-1)/(i<<1));
for(int j=0;j<n;j+=i<<1)
{
int w=1;
for(int k=j;k<i+j;k++,w=(ll)W*w%mod)
{
int x=p[k],y=(ll)w*p[k+i]%mod;
p[k]=x+y;p[k+i]=x-y;
if(p[k]>=mod)p[k]-=mod;
if(p[k+i]<0)p[k+i]+=mod;
}
}
}
if(op==-1)
{
int inv=power(n,mod-2);
REP(i,0,n-1)p[i]=(ll)p[i]*inv%mod;
reverse(p+1,p+n);
}
}
void Inv(int *p,int *q,int len)
{
if(len==1){ q[0]=power(p[0],mod-2);return;}
Inv(p,q,len>>1);
REP(i,0,len-1)A[i]=p[i],B[i]=q[i];
NTT(A,len<<1,1);NTT(B,len<<1,1);
REP(i,0,(len<<1)-1)A[i]=(ll)A[i]*B[i]%mod*B[i]%mod;
NTT(A,len<<1,-1);
REP(i,0,len-1)q[i]=((2*q[i]-A[i])%mod+mod)%mod;//len不需要左移
REP(i,0,(len<<1)-1)A[i]=B[i]=0;
}
inline void doing()
{
int N=1,l=0;
REP(i,1,n)g[i]=(-jcn[i]*2+2*mod)%mod;
g[0]=1;
while(N<=n)N<<=1,l++;
Inv(g,ans,N);
int Ans=0;
REP(i,0,n)Ans=(Ans+(ll)ans[i]*jc[i])%mod;
printf("%d\n",Ans);
}
int main()
{
init();
doing();
return 0;
}
解法三
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<cmath>
#include<queue>
#include<stack>
#include<set>
#include<map>
using namespace std;
#define ll long long
#define REP(i,a,b) for(int i=(a),_end_=(b);i<=_end_;i++)
#define DREP(i,a,b) for(int i=(a),_end_=(b);i>=_end_;i--)
#define EREP(i,a) for(int i=start[(a)];i;i=e[i].next)
inline int read()
{
int sum=0,p=1;char ch=getchar();
while(!(('0'<=ch && ch<='9') || ch=='-'))ch=getchar();
if(ch=='-')p=-1,ch=getchar();
while('0'<=ch && ch<='9')sum=sum*10+ch-48,ch=getchar();
return sum*p;
}
const int maxn=4e5+20;
const int mod=998244353;
int n,inv[maxn],jc[maxn],jcn[maxn];
inline int power(int a,int b)
{
int ans=1;
while(b)
{
if(b & 1)ans=(ll)ans*a%mod;
b>>=1;
a=(ll)a*a%mod;
}
return ans;
}
int f[maxn],g[maxn],rev[maxn];
inline void NTT(int *p,int n,int opt)
{
REP(i,0,n-1)if(i<rev[i])swap(p[i],p[rev[i]]);
for(int i=1;i<n;i<<=1)
{
int W=power(3,(mod-1)/(i<<1));
for(int j=0;j<n;j+=i<<1)
{
int w=1;
for(int k=j;k<i+j;k++,w=(ll)w*W%mod)
{
int x=p[k],y=(ll)p[k+i]*w%mod;
p[k]=x+y;
p[k+i]=x-y;
if(p[k]>=mod)p[k]-=mod;
if(p[k+i]<0)p[k+i]+=mod;
}
}
}
if(opt==-1)
{
int inv=power(n,mod-2);
REP(i,0,n-1)p[i]=(ll)p[i]*inv%mod;
reverse(p+1,p+n);
}
}
void init()
{
n=read();inv[1]=jc[0]=jc[1]=jcn[0]=jcn[1]=1;
REP(i,2,n+1)jc[i]=(ll)i*jc[i-1]%mod,inv[i]=(ll)(mod-mod/i)*inv[mod%i]%mod,jcn[i]=(ll)inv[i]*jcn[i-1]%mod;
f[0]=g[0]=1;g[1]=n+1;
REP(i,1,n)f[i]=(ll)(mod-inv[i])*f[i-1]%mod;
REP(i,2,n)g[i]=(ll)(power(i,n+1)-1)*inv[i-1]%mod*jcn[i]%mod;
int N=1,l=0;while(N<=2*n)N<<=1,l++;
REP(i,1,N-1)rev[i]=(rev[i>>1]>>1)|((i&1)<<(l-1));
NTT(f,N,1);
NTT(g,N,1);
REP(i,0,N-1)f[i]=(ll)f[i]*g[i]%mod;
NTT(f,N,-1);
}
void doing()
{
int ans=0,bin=1;
REP(i,0,n)ans=(ans+(ll)f[i]*jc[i]%mod*bin)%mod,bin=(bin<<1)%mod;
printf("%d\n",ans);
}
int main()
{
init();
doing();
return 0;
}
[HEOI2016]求和 sum的更多相关文章
- BZOJ 4555: [Tjoi2016&Heoi2016]求和 [分治FFT 组合计数 | 多项式求逆]
4555: [Tjoi2016&Heoi2016]求和 题意:求\[ \sum_{i=0}^n \sum_{j=0}^i S(i,j)\cdot 2^j\cdot j! \\ S是第二类斯特林 ...
- BZOJ 4555: [Tjoi2016&Heoi2016]求和 [FFT 组合计数 容斥原理]
4555: [Tjoi2016&Heoi2016]求和 题意:求\[ \sum_{i=0}^n \sum_{j=0}^i S(i,j)\cdot 2^j\cdot j! \\ S是第二类斯特林 ...
- 【BZOJ】4555: [Tjoi2016&Heoi2016]求和 排列组合+多项式求逆 或 斯特林数+NTT
[题意]给定n,求Σi=0~nΣj=1~i s(i,j)*2^j*j!,n<=10^5. [算法]生成函数+排列组合+多项式求逆 [题解]参考: [BZOJ4555][Tjoi2016& ...
- 【BZOJ 4555】 4555: [Tjoi2016&Heoi2016]求和 (NTT)
4555: [Tjoi2016&Heoi2016]求和 Time Limit: 40 Sec Memory Limit: 128 MBSubmit: 315 Solved: 252 Des ...
- [BZOJ4555][TJOI2016&HEOI2016]求和(分治FFT)
4555: [Tjoi2016&Heoi2016]求和 Time Limit: 40 Sec Memory Limit: 128 MBSubmit: 525 Solved: 418[Sub ...
- bzoj 4555 [Tjoi2016&Heoi2016]求和 NTT 第二类斯特林数 等比数列求和优化
[Tjoi2016&Heoi2016]求和 Time Limit: 40 Sec Memory Limit: 128 MBSubmit: 679 Solved: 534[Submit][S ...
- [BZOJ4555 TJOI2016 HEOI2016 求和]
第一篇博客,请大家多多关照.(鞠躬 BZOJ4555 TJOI2016 HEOI2016 求和 题意: 给定一个正整数\(n\)(\(1\leqq n \leqq100000\)),求: \[ ...
- BZOJ 4555: [Tjoi2016&Heoi2016]求和 (NTT + 第二类斯特林数)
题意 给你一个数 \(n\) 求这样一个函数的值 : \[\displaystyle f(n)=\sum_{i=0}^{n}\sum_{j=0}^{i} \begin{Bmatrix} i \\ j ...
- [BZOJ 4555][Tjoi2016&Heoi2016]求和
题意 给定 $n$ , 求下式的值: $$ f(n)= \sum_{i=0}^n\sum_{j=0}^i\begin{Bmatrix}i\\ j\end{Bmatrix}\times 2^j\time ...
随机推荐
- mysql修改表和列
mysql修改列 mysql增加列,修改列名.列属性,删除列语句 mysql修改表名,列名,列类型,添加表列,删除表列 alter table test rename test1; --修 ...
- Python实现一个简单的微信跳一跳辅助
1. 前言 微信的跳一跳相信大家都很熟悉了,而且现在各种外挂.辅助也是满天飞,反正本人的好友排行榜中已经是八九百都不足为奇了.某宝上一搜一堆结果,最低的居然只要3块多,想刷多少分就刷多少分,真是离谱 ...
- Mysql索引分析:适合建索引?不适合建索引?【转】
数据库建立索引常用的规则如下: 1.表的主键.外键必须有索引: 2.数据量超过300的表应该有索引: 3.经常与其他表进行连接的表,在连接字段上应该建立索引: 4.经常出现在Where子句中的字段,特 ...
- 我的Android手册
目录解释说明 assets文件说明 app_id:机智云 app id app_secret:机智云 app secret product_key:机智云 product key wifi_type_ ...
- 防盗链[referer]
原文出处:http://www.cnblogs.com/devilfree/archive/2012/09/11/2680914.html 总结一下今天学习防盗链Filter的一些知识点: 防盗链要实 ...
- 三、Html常用标签
1,基本标签 <html>:html文档的根元素,可以指定一个xmlns属性,值只能是http://www/w3.org/1999/xhtml. <body>:页面主体部分 & ...
- ueditor显示内容末尾有多余标记的解决
问题: 最近用了百度的ueditor文本编辑器,出现一个问题,用ueditor存数据到数据库都正常,但是重新读取后赋值到ueditor却会在末尾多出 "> 这两个符号.赋值方式如下: ...
- Spring事务不回滚原因分析
Synchronized用于线程间的数据共享,而ThreadLocal则用于线程间的数据隔离. 在我完成一个项目的时候,遇到了一个Spring事务不回滚的问题,通过aspectJ和@Transacti ...
- vs调试dll工程
dll本身是没法运行的,必须在其它工程调用dll时候才会运行. 所以,调试dll首先要将调用dll的工程和dll工程联系起来. 解决方案中添加dll工程: 现在dll 和 应用程序两个工程就都在一个解 ...
- Titanic数据分析
一.材料准备 https://www.kaggle.com/c/titanic-gettingStarted/ 二.提出问题 生存率和哪些因素有关(性别,年龄,是否有伴侣,票价,舱位等级,包间,出发地 ...