【Topcoder SRM697 Hard】【JZOJ5180】ConnectedStates 题解

题目大意

  $n \le 2000,\ w_i \le 10^9$

题解

  稍微有点妙啊。。。

  与度数有关的无根树计数,考虑 prufer 序。
  暴力可以直接 $O(n^3)$ dp 计算答案。

  考虑优化。

  假设第 $i$ 个数在 prufer 序的出现次数是 $a_i$,那么求的是:

  其中与 $a$ 无关的是 $(n-2)!×\prod w_i$,去掉之后原式变成:

  接着考虑拆开 $\prod (a_i+1)$,拆开后的每一项就相当于我选择一些 $a_i$ 乘起来。假设我选择的是 $a_{p_1},a_{p_2},…,a_{p_k}$,则有:

  上面的 $a$ 会跟下面的阶乘约掉,那么我可以一开始就给这一部分 $a$ 减 $1$,然后乘上后面少了的 $w$,相当于:

  注意到我只要枚举 $k$ 的话,前后两部分就独立了。前面是个背包,所以现在化简后面。

  后面这个东西跟 EGF(指数型生成函数) 很像,相当于求 :

  于是就。。做完了。

代码

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
#include<cstdio>
#define fo(i,a,b) for(int i=a;i<=b;i++)
#define fd(i,a,b) for(int i=a;i>=b;i--)
using namespace std;

typedef long long LL;

const int maxn=2005;
const LL mo=1e9+7;

int n,w[maxn];
LL sumw,prow=1;

LL fac[maxn],ny[maxn],f[maxn][maxn];
LL mi(LL x,LL y)
{
LL re=1;
for(; y; y>>=1, x=x*x%mo) if (y&1) re=re*x%mo;
return re;
}
void Pre()
{
fac[0]=ny[0]=1;
fo(i,1,n) fac[i]=fac[i-1]*i%mo;
ny[n]=mi(fac[n],mo-2);
fd(i,n-1,1) ny[i]=ny[i+1]*(i+1)%mo;

f[0][0]=1;
fo(i,1,n)
fo(j,0,i)
{
f[i][j]=f[i-1][j];
if (j) (f[i][j]+=f[i-1][j-1]*w[i])%=mo;
}
}

int main()
{
scanf("%d",&n);
fo(i,1,n)
{
scanf("%d",&w[i]);
if (!w[i]) {printf("0\n"); return 0;}
(sumw+=w[i])%=mo;
(prow*=w[i])%=mo;
}

Pre();

LL ans=0;
fo(k,0,n-2) (ans+=f[n][k]*mi(sumw,n-2-k)%mo*ny[n-2-k])%=mo;

printf("%lld\n",ans*prow%mo*fac[n-2]%mo);
}