很久以前写过的一道题。
今天突然发现题解全是生成函数,就来补一篇正常一点的。
题目链接。
题意
现在有  个盒子,第  个盒子初始有  个小球。
现在要进行 
次操作,每一次随机向一个盒子放一个小球。
设最终第  个盒子里有  个小球,求  的期望。
。
。
题解
首先设 。
那么 。
考虑把这个式子暴力拆开。
那么就是对于每一个 ,选出  和  中的一个,然后乘起来,最后加和。
设  为前  个数中选出  个 ,乘积的和是多少。
那么有转移 。
这个时候 
就表示所有数中选出  个 ,乘积的和是多少。
这个时候我们只需要求出,剩下 
个  期望乘积是多少。
首先发现,任意  个  期望乘积都是一样的。
所以我们只需要求出  表示取
 个 ,期望乘积是多少,这里不妨假设是前  个盒子。
设  表示第  个操作,有没有向第  个盒子投球。
那么 ,考虑继续拆期望。  直接计算每一项,单项是  的。
最终答案就是 。
时间复杂度 。
代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62
   |  #include<algorithm> #include<cstring> #include<cstdio> #include<vector> #include<queue> #include<map> #define inf 0x3f3f3f3f3f3f3f3fll using namespace std; const int mod=998244353; using ll=long long; using pli=pair<ll,int>; using pi=pair<int,int>; template<typename A> using vc=vector<A>; inline int read() {     int s=0,w=1;char ch;     while((ch=getchar())>'9'||ch<'0') if(ch=='-') w=-1;     while(ch>='0'&&ch<='9') s=s*10+ch-'0',ch=getchar();     return s*w; } ll dp[1001][1001]; int a[1001]; int n,k; ll ans; inline ll qow(ll a,int b) {     ll ans=1;     while(b)     {         if(b&1) ans=ans*a%mod;         a=a*a%mod;         b>>=1;     }     return ans; } inline ll E(int n) {     ll val=1;     for(int i=0;i<n;i++) val=val*(k-i)%mod;          return val*qow(qow(::n,n),mod-2)%mod; } int main() {     n=read(),k=read();     for(int i=1;i<=n;i++) a[i]=read()%mod;     dp[0][0]=1;     for(int i=1;i<=n;i++)     {         dp[i][0]=1;         for(int j=1;j<=i;j++) dp[i][j]=(dp[i-1][j]+dp[i-1][j-1]*a[i])%mod;     }     for(int i=0;i<=n;i++)     {                  ans=(ans+dp[n][i]*E(n-i))%mod;     }     printf("%lld\n",ans);     return 0; }
 
  | 
 
感谢观看!