定義一個序列的權值爲:把全部相鄰的相同的數合併爲一個集合後,全部集合的大小的乘積。spa
特別的,第一個數和最後一個數是相鄰的。code
如今你有 \(n\) 種數,第 \(i\) 種有 \(c_i\) 個。求全部不一樣的序列的權值的和。string
\(n\leq 50,c_i\leq 100\)it
考慮第一個數和最後一個數不相鄰時怎麼作。io
記 \(g_{i,j}\) 爲出現了 \(i\) 次的數分紅 \(j\) 個集合,全部集合大小的乘積的和。
\[ g_{i,j}=\sum_{k=1}^ig_{i-k,j-1}\times k \]
假設最後 \(i\) 分紅了 \(a_i\) 個集合,那麼答案就是 \(\prod_{i=1}^ng_{c_i,a_i}\) 再乘上方案數。class
方案數能夠容斥求。im
具體來講,把最後相鄰且同色的球合併成一個大球。設最後有 \(b_i\) 個大球,那麼容斥係數就是 \({(-1)}^{a_i-b_i}\),帶容斥係數的方案數就是 \(\binom{a_i-1}{b_i-1}{(-1)}^{a_i-b_i}\)集合
最後這 \(\sum b_i\) 個球能夠隨意放,方案數是 \(\frac{(\sum b_i)!}{\prod b_i!}\)di
總的答案是
\[ \left(\prod_{i=1}^ng_{c_i,a_i}\binom{a_i-1}{b_i-1}{(-1)}^{a_i-b_i}\right)\frac{(\sum_{i=1}^nb_i)!}{\prod_{i=1}^n b_i!} \]
這樣就能夠 DP 了。(狀態爲 \(i\) 和 \(\sum b_i\))時間
考慮第一個數和最後一個數相鄰時怎麼作。
能夠用最小表示法,令第一個數爲 \(1\) 且 最後一個數不爲 \(1\)(除非 \(n=1\))。
只須要在後面計算組合數的時候把 \(b_1-1\) 再除以 \(a_1\) 就能夠獲得第一個數爲 \(1\) 的方案數。
把 \(b_1-2\) 再除以 \(a_1\) 就能夠獲得第一個數爲 \(1\) 且最後一個數也是 \(1\) 的方案數。
除以 \(a_1\) 是由於一個方案會被算屢次。
再把方案數乘以 \(\sum c_i\) 就是答案了。
時間複雜度:\(O((\sum c_i)^2)\)
#include<cstdio> #include<cstring> #include<algorithm> using namespace std; typedef long long ll; const ll p=1000000007; ll fac[5010],ifac[5010],inv[5010]; ll f[60][5010]; ll g[110][110]; int a[60]; int n; int s[60]; ll c[110][110]; ll c1[110],c2[110]; ll binom(int x,int y) { return x>=y&&y>=0?fac[x]*ifac[y]%p*ifac[x-y]%p:0; } int main() { #ifndef ONLINE_JUDGE freopen("c.in","r",stdin); freopen("c.out","w",stdout); #endif scanf("%d",&n); for(int i=1;i<=n;i++) { scanf("%d",&a[i]); s[i]=s[i-1]+a[i]; } inv[1]=fac[0]=fac[1]=ifac[0]=ifac[1]=1; for(int i=2;i<=5000;i++) { inv[i]=-p/i*inv[p%i]%p; fac[i]=fac[i-1]*i%p; ifac[i]=ifac[i-1]*inv[i]%p; } g[0][0]=1; for(int i=1;i<=100;i++) for(int j=1;j<=100;j++) for(int k=1;k<=i;k++) g[i][j]=(g[i][j]+g[i-k][j-1]*k)%p; f[0][0]=1; for(int i=1;i<n;i++) for(int j=1;j<=a[i];j++) for(int k=1;k<=j;k++) c[i][k]=(c[i][k]+g[a[i]][j]*binom(j-1,k-1)%p*((j-k)&1?-1:1))%p; for(int i=n;i<=n;i++) for(int j=1;j<=a[i];j++) for(int k=1;k<=j;k++) c1[k]=(c1[k]+g[a[i]][j]*binom(j-1,k-1)%p*((j-k)&1?-1:1)*inv[j])%p; for(int i=1;i<=n;i++) for(int j=1;j<=a[i];j++) for(int k=0;k<=s[i-1];k++) f[i][k+j]=(f[i][k+j]+f[i-1][k]*c[i][j]%p*binom(k+j,k))%p; ll ans=0; for(int j=1;j<=a[n];j++) for(int k=0;k<=s[n-1];k++) ans=(ans+f[n-1][k]*c1[j]%p*(binom(k+j-1,k)-binom(k+j-2,k)))%p; ans=ans*s[n]%p; ans=(ans+p)%p; printf("%lld\n",ans); return 0; }