ABC279H 题解

学考时候推的。场上记错模数,以为是 \(998244353\) 然后想了半天组合数怎么算)

简单题。

首先序列问题考虑每个位置的生成函数然后乘起来。位置 \(k\) 的生成函数显然是

\[\sum_{i=1}\min(i,k)x^i \]

那么观察一下尝试搞封闭形式。它前 \(k\) 项和 \(\dfrac x{(1-x)^2}\) 是一样的,后边全是 \(k\)。我的推法是大减小,即:

\[\begin{aligned} &\sum_{i=1}\min(i,k)x^i\\ =&\sum_{i=1}ix^i-\sum_{i=0}ix^{i+k}\\ =&\frac x{(1-x)^2}-x^k\frac x{(1-x)^2}\\ =&x\frac {1-x^k}{(1-x)^2} \end{aligned} \]

那么答案就是

\[\begin{aligned} &[x^m]\prod_{i=1}^nx\frac{1-x^i}{(1-x)^2}\\ =&[x^{m-n}]\prod_{i=1}^n\frac{1-x^i}{(1-x)^2}\\ =&[x^{m-n}]\frac 1{(1-x)^{2n}}\prod_{i=1}^n(1-x^i) \end{aligned} \]

前边这个东西的第 \(k\) 项显然是 \(\dbinom{2n-1+k}k\)。关于后边这个东西,一看长的就很像五边形数。由于 \(m-n\le n\),那么 \([x^{m-n}]\prod_{i=1}^n(1-x^i)=[x^{m-n}]\prod_{i=1}(1-x^i)=[x^{m-n}]1+\sum_{i=1}(-1)^ix^{\frac{i(3i\pm 1)}2}\)。只有 \(O(\sqrt n)\) 个取值,lucas 爆算即可。

#include <cstdio>
#include <iostream>
#include <algorithm>
#include <cstring>
#define int long long
using namespace std;
const int mod=200003;
int n,m,ans,jc[200010],inv[200010];
int binom(int n,int m){
    if(m<0||n<m)return 0;
    return 1ll*jc[n]*inv[m]%mod*inv[n-m]%mod;
}
int C(int n,int m){
    if(!m)return 1;
    return 1ll*binom(n%mod,m%mod)*C(n/mod,m/mod)%mod;
}
signed main(){
    scanf("%lld%lld",&n,&m);jc[0]=inv[0]=inv[1]=1;
    for(int i=1;i<mod;i++)jc[i]=1ll*jc[i-1]*i%mod;
    for(int i=2;i<mod;i++)inv[i]=1ll*(mod-mod/i)*inv[mod%i]%mod;
    for(int i=1;i<mod;i++)inv[i]=1ll*inv[i-1]*inv[i]%mod;
    for(int i=0;;i++){
        int ret=i*(3*i-1)>>1;
        if(ret>m-n)break;
        ans=(ans+1ll*((i&1)?mod-1:1)*C(m+n-ret-1,m-n-ret))%mod;
        ret=i*(3*i+1)>>1;
        if(!i)continue;
        if(ret>m-n)break;
        ans=(ans+1ll*((i&1)?mod-1:1)*C(m+n-ret-1,m-n-ret))%mod;
    }
    printf("%d\n",ans);
    return 0;
}
posted @ 2023-03-07 20:47  gtm1514  阅读(25)  评论(0编辑  收藏  举报