题目描述

给一棵树,求以每个点为根时下列式子的值。

Crash 的文明世界-LMLPHP

题解

当k=1时这就是一个经典的换根dp问题。

所以这道题还是要用换根dp解决。

部分分做法:

考虑转移时是这样的一个形式(图是抄的)。

Crash 的文明世界-LMLPHP

Crash 的文明世界-LMLPHP

用二项式定理展开就可以nk做了。

观察到结果是一个x的形式。

然后这个可以用斯特林数代换。

Crash 的文明世界-LMLPHP

我们可以先求出每个点的后面的东西,在乘上前面的就是答案了。

这是个组合数,可以用组合数的递推解决。

Crash 的文明世界-LMLPHP

代码

#include<iostream>
#include<cstdio>
#define N 50009
#define KK 151
using namespace std;
typedef long long ll;
const int mod=;
int dp[N][KK],f[KK],h[KK],jie[KK];
int n,m,a[N],tot,head[N],K,s[KK][KK];
inline ll rd(){
ll x=;char c=getchar();bool f=;
while(!isdigit(c)){if(c=='-')f=;c=getchar();}
while(isdigit(c)){x=(x<<)+(x<<)+(c^);c=getchar();}
return f?-x:x;
}
struct edge{int n,to;}e[N<<];
inline void add(int u,int v){
e[++tot].n=head[u];e[tot].to=v;head[u]=tot;
e[++tot].n=head[v];e[tot].to=u;head[v]=tot;
}
void dfs(int u,int fa){
dp[u][]=;
for(int i=head[u];i;i=e[i].n)if(e[i].to!=fa){
int v=e[i].to;dfs(v,u);
(dp[u][]+=dp[v][])%=mod;
for(int j=;j<=K;++j)(dp[u][j]+=dp[v][j]+dp[v][j-])%=mod;
}
}
void dfs2(int u,int fa){
for(int i=head[u];i;i=e[i].n)if(e[i].to!=fa){
int v=e[i].to;
for(int j=;j<=K;++j)f[j]=;
f[]=dp[u][]-dp[v][];
for(int j=;j<=K;++j)(f[j]+=dp[u][j]-dp[v][j-]-dp[v][j]+mod*)%=mod;
(dp[v][]+=f[])%=mod;
for(int j=;j<=K;++j)(dp[v][j]+=f[j]+f[j-])%=mod;
dfs2(v,u);
}
}
int main(){
n=rd();K=rd();int u,v;
for(int i=;i<n;++i){u=rd();v=rd();add(u,v);}
s[][]=;
for(int i=;i<=K;++i){
s[i][]=;
for(int j=;j<=i;++j)
s[i][j]=(s[i-][j-]+s[i-][j]*j)%mod;
}
jie[]=;
for(int i=;i<=K;++i)jie[i]=jie[i-]*i%mod;
dfs(,);
dfs2(,);
for(int i=;i<=n;++i){
int ans=;
for(int j=;j<=K;++j)(ans+=s[K][j]*jie[j]%mod*dp[i][j]%mod)%=mod;
printf("%d\n",ans);
}
return ;
}
05-11 20:03