bzoj4555 [HEOI2016/TJOI2016]求和

13 篇文章 0 订阅

题目描述


在2016年,佳媛姐姐刚刚学习了第二类斯特林数,非常开心。
现在他想计算这样一个函数的值:

f(n)=i=0nj=0iS(i,j)×j!×2j f ( n ) = ∑ i = 0 n ∑ j = 0 i S ( i , j ) × j ! × 2 j

S(i, j)表示第二类斯特林数
你能帮帮他吗?

输出f(n)。由于结果会很大,输出f(n)对998244353(7 × 17 × 223 + 1)取模的结果即可

Solution


首先j的枚举上界可以直接变成n,原因可以考虑当i< j时第二类斯特林数为0
第二类斯特林数可以拆,那么就是

i=0nj=0n2j×j!k=0j(1)k×C(j,k)×(jk)i ∑ i = 0 n ∑ j = 0 n 2 j × j ! ∑ k = 0 j ( − 1 ) k × C ( j , k ) × ( j − k ) i

组合数也可以拆,那么就是
i=0nj=0n2j×j!k=0j(1)kk!×(jk)i(jk)! ∑ i = 0 n ∑ j = 0 n 2 j × j ! ∑ k = 0 j ( − 1 ) k k ! × ( j − k ) i ( j − k ) !

把最外面的∑拉到最里面有
j=0n2j×j!k=0j(1)kk!×ni=0(jk)i(jk)! ∑ j = 0 n 2 j × j ! ∑ k = 0 j ( − 1 ) k k ! × ∑ i = 0 n ( j − k ) i ( j − k ) !

后半部分实际上是个卷积,i的∑实际上是等比数列求和,NTT一下求解就好了

Code


#include <stdio.h>
#include <string.h>
#include <algorithm>
#include <vector>
#define rep(i,st,ed) for (int i=st;i<=ed;++i)

typedef long long LL;
const int MOD=998244353;
const int N=300005;
const int G=3;

int fac[N],a[N],b[N],c[N];
int rev[N],inv[N];
int len,lg,ny;

LL ksm(LL x,LL dep) {
    if (dep==0) return 1;
    if (dep==1) return x;
    LL tmp=ksm(x,dep/2);
    if (dep&1) return tmp*tmp%MOD*x%MOD;
    return tmp*tmp%MOD;
}

void DFT(int *a,int f) {
    rep(i,0,len-1) if (i<rev[i]) std:: swap(a[rev[i]],a[i]);
    for (int i=1;i<len;i*=2) {
        int wn;
        if (f==1) wn=ksm(G,(MOD-1)/i/2);
        else wn=ksm(G,MOD-1-(MOD-1)/i/2);
        for (int j=0;j<len;j+=i*2) {
            int w=1;
            for (int k=0;k<i;k++) {
                int u=a[j+k],v=(LL)a[j+k+i]*w%MOD;
                a[j+k]=(u+v)%MOD;
                a[j+k+i]=(u-v)%MOD;
                w=(LL)w*wn%MOD;
            }
        }
    }
    if (f==-1) rep(i,0,len-1) a[i]=(LL)a[i]*ny%MOD;
}

void pre_work(int n) {
    fac[0]=1; rep(i,1,n) fac[i]=(LL)fac[i-1]*i%MOD;
    inv[0]=1; rep(i,1,n) inv[i]=ksm(fac[i],MOD-2);
    for (len=1;len<=n*2;len<<=1,lg++);
    rep(i,0,len-1) rev[i]=(rev[i/2]/2)|((i&1)<<(lg-1));
    ny=ksm(len,MOD-2);
}

int main(void) {
    int n; scanf("%d",&n); pre_work(n);
    rep(i,0,n) a[i]=((i&1)?(-1):(1))*inv[i];
    a[0]=1; b[0]=1; b[1]=n+1;
    rep(i,2,n) b[i]=(LL)((LL)(ksm(i,n+1)-1)*inv[i]%MOD*ksm(i-1,MOD-2)%MOD+MOD)%MOD;
    rep(i,0,n) printf("%d %d\n", a[i],b[i]);
    DFT(a,1); DFT(b,1);
    rep(i,0,len-1) a[i]=(LL)a[i]*b[i]%MOD;
    DFT(a,-1);
    LL ans=0; rep(i,0,n) ans=(ans+(LL)a[i]*fac[i]%MOD*ksm(2,i)%MOD)%MOD;
    printf("%lld\n", (ans+MOD)%MOD);
    return 0;
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值