题面

解析

这题一眼换根DP啊

首先,我们考虑一下如何转换\(n^m\)这个式子,

先把式子摆出来吧:\(n^m=\sum_{j=0}^mS(m,j)C_n^jj!\)

其中\(S(m,j)\)表示第二类斯特林数,

(其实就是把\(m\)个不同的小球放到\(j\)个相同的盒子里,且盒子不为空的方案数)

我们感性理解一下(懒得不会推式子),

\(n^m\)其实就表示将\(m\)个不同的球放入\(n\)个不同的盒子里,

而我们可以枚举有\(j\)个盒子里面放了小球,

那么方案数就是\(S(m,j)\)(如上)\(*C_n^j\)(选出\(j\)个盒子的方案数)\(*j!\)(因为盒子是不同的所以有排列)

那么上面的式子就很好理解了吧.

然而这到底有什么用?

我们将题目中的条件代入到上式中,

对于每个节点\(x\),

我们要求的是\(\sum_{i=1}^n dis(x,i)^k\),其中\(dis(i,j)\)表示\(i,j\)的距离,

首先考虑\(x\)为根,

那么\(dis(x,i)\)也就是\(dep[i]\)(深度)(\(dep[x]=0\)).

而式子就可以变形成\(\sum_{i=1}^n\sum_{j=0}^kS(k,j)C_{dep[i]}^jj!\)

\(=\sum_{j=0}^k\sum_{i=1}^nS(k,j)j!C_{dep[i]}^j\)

\(=\sum_{j=0}^kS(k,j)j!\sum_{i=1}^nC_{dep[i]}^j\)

所以我们可以发现,

真正与节点有关的,就是\(\sum_{i=1}^nC_{dep[i]}^j\),

而前面的都是常数.

所以,我们设\(f[x][j]\)表示\(\sum_iC_{dep[i]}^j\),\(i\)为以\(x\)为根的子树中的节点,

而根据\(C_n^m=C_{n-1}^m+C_{n-1}^{m-1}\),

我们可以得到\(f[x][j]=\sum_iC_{dep[i]-1}^j+C_{dep[i]-1}^{j-1}\),

而对于\(x\)的子节点\(i\)来说,在\(i\)的子树中,每个点的深度对比以\(x\)为根都减了一,

因此,递推式就能出来了:

\(f[x][j]=\sum_{i=son[x]}f[i][j]+f[i][j-1]\).

然而,这个式子只对于一开始钦定的整棵树的根有效,

因此我们还需要来一次换根DP,

这个看代码就能理解了:

#include <iostream>
#include <cstdio>
#include <cstring>
#define ll long long
#define fre(x) freopen(x".in","r",stdin),freopen(x".out","w",stdout)
using namespace std; inline int read(){
int sum=0,f=1;char ch=getchar();
while(ch>'9' || ch<'0'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0' && ch<='9'){sum=sum*10+ch-'0';ch=getchar();}
return f*sum;
} const int N=100001;
const int Mod=10007;
struct edge{int to,next;}e[N<<1];
int n,K;
int head[N],cnt;
int jc[201],s[201][201];
int f[N][201],g[N][201],tem[201]; inline void add(int x,int y){
e[++cnt]=(edge){head[x],y};head[x]=cnt;
} inline void dfs(int x,int fa){
f[x][0]=1;
for(int i=head[x];i;i=e[i].to){
int k=e[i].next;if(k==fa) continue;
dfs(k,x);f[x][0]=(f[x][0]+f[k][0])%Mod;
for(int j=1;j<=K;j++) f[x][j]=(f[x][j]+f[k][j]+f[k][j-1])%Mod;
}
} inline void dfs2(int x,int fa){
for(int j=0;j<=K;j++) g[x][j]=f[x][j];
if(fa){
for(int j=1;j<=K;j++) tem[j]=(g[fa][j]-f[x][j]+Mod-f[x][j-1]+Mod)%Mod;
tem[0]=(g[fa][0]-f[x][0]+Mod)%Mod;
for(int j=1;j<=K;j++) g[x][j]=(g[x][j]+tem[j]+tem[j-1])%Mod;
g[x][0]=(g[x][0]+tem[0])%Mod;
}
for(int i=head[x];i;i=e[i].to){
int k=e[i].next;if(k==fa) continue;
dfs2(k,x);
}
} int main(){
n=read();K=read();
for(int i=1;i<n;i++){
int x=read(),y=read();
add(x,y);add(y,x);
}
jc[0]=1;s[1][1]=s[0][0]=1;
for(int i=1;i<=K;i++) jc[i]=jc[i-1]*i%Mod;
for(int i=1;i<=K;i++)
for(int j=1;j<=K;j++)
s[i][j]=(s[i-1][j]*j+s[i-1][j-1])%Mod;//预处理
dfs(1,0);/*求出f[x][j]*/dfs2(1,0);/*换根DP*/
for(int i=1;i<=n;i++){
ll ans=0;
for(int j=0;j<=K;j++) ans=(ans+s[K][j]*jc[j]%Mod*g[i][j]%Mod)%Mod;
printf("%lld\n",(ans+Mod)%Mod);
}
return 0;
}

题解 [BZOJ2159] Crash的文明世界的更多相关文章

  1. BZOJ2159 Crash 的文明世界 【第二类斯特林数 + 树形dp】

    题目链接 BZOJ2159 题解 显然不能直接做点分之类的,观察式子中存在式子\(n^k\) 可以考虑到 \[n^k = \sum\limits_{i = 0} \begin{Bmatrix} k \ ...

  2. BZOJ2159 Crash的文明世界(树形dp+斯特林数)

    根据组合意义,有nk=ΣC(n,i)*i!*S(k,i) (i=0~k),即将k个有标号球放进n个有标号盒子的方案数=在n个盒子中选i个将k个有标号球放入并且每个盒子至少有一个球. 回到本题,可以令f ...

  3. [BZOJ2159]Crash的文明世界(斯特林数+树形DP)

    题意:给定一棵树,求$S(i)=\sum_{j=1}^{n}dist(i,j)^k$.题解:根据斯特林数反演得到:$n^m=\sum_{i=0}^{n}C(n,i)\times i!\times S( ...

  4. BZOJ2159 Crash的文明世界

    Description 传送门 给你一个n个点的树,边权为1. 对于每个点u, 求:\(\sum_{i = 1}^{n} distance(u, i)^{k}\) $ n \leq 50000, k ...

  5. BZOJ2159 : Crash 的文明世界

    $x^k=\sum_{i=1}^k Stirling2(k,i)\times i!\times C(x,i)$ 设$f[i][j]=\sum_{k=1}^n C(dist(i,k),j)$. 则可以利 ...

  6. BZOJ2159 Crash的文明世界——树上DP&&第二类Stirling数

    题意 给定一个有 $n$ 个结点的树,设 $S(i)$ 为第 $i$ 个结点的“指标值”,定义为 $S(i)=\sum_{i=1}^{n}dist(i,j)^k$,$dist(i, j)$ 为结点 $ ...

  7. 【BZOJ2159】Crash的文明世界(第二类斯特林数,动态规划)

    [BZOJ2159]Crash的文明世界(第二类斯特林数,动态规划) 题面 BZOJ 洛谷 题解 看到\(k\)次方的式子就可以往二项式的展开上面考,但是显然这样子的复杂度会有一个\(O(k^2)\) ...

  8. 【BZOJ2159】Crash的文明世界

    [2011集训贾志鹏]Crash的文明世界 Description Crash小朋友最近迷上了一款游戏--文明5(Civilization V).在这个游戏中,玩家可以建立和发展自己的国家,通过外交和 ...

  9. P4827「国家集训队」 Crash 的文明世界

    「国家集训队」 Crash 的文明世界 提供一种不需要脑子的方法. 其实是看洛谷讨论版看出来的( (但是全网也就这一篇这个方法的题解了) 首先这是一个关于树上路径的问题,我们可以无脑上点分治. 考虑当 ...

随机推荐

  1. LeetCode 第 165 场周赛

    LeetCode 第 165 场周赛 5275. 找出井字棋的获胜者 5276. 不浪费原料的汉堡制作方案 5277. 统计全为 1 的正方形子矩阵 5278. 分割回文串 III C 暴力做的,只能 ...

  2. CodeBlocks 配置

    CodeBlocks 配置 Code::Blocks 17.12 时间:2019.6 下载网址 http://www.codeblocks.org/downloads/26 ,这里选择的是 mingw ...

  3. 剑指offer1: 组类型——二维数组中的查找(给定一个数字,查找是否在该数组中)

    1. 思路: 缩小范围 2. 方法: (1)要查找的数字等于数组中的数字,结束查找过程: (2)要查找的数字小于数组中的数字,去除该数字右边的数字,在剩下的数字里查找: (3)要查找的数字大于数组中的 ...

  4. 最大流Dinic(模板)

    #define IOS ios_base::sync_with_stdio(0); cin.tie(0); #include <cstdio>//sprintf islower isupp ...

  5. python学习-8 用户有三次机会登陆

    用户登陆(三次机会) count = 0 while count < 3: user = input('请输入账号:') pwd = input('请输入密码:') ': print(" ...

  6. SpringBoot事务隔离等级和传播行为

    一.开启事物管理 //import org.springframework.transaction.annotation.EnableTransactionManagement; @SpringBoo ...

  7. android 自定义控件之NetWorkImageView 处理listview等控件中的图片加载乱序问题

    0.调用: BaseAdapter中设置方法 holder.iv.loadImage(url); adapter_xxx.xml 中 控件需要用 xxx.NetWorkImageView 1 NetW ...

  8. Web API 接口版本控制 SDammann.WebApi.Versioning

    前言 在设计对外 Web API 时,实务上可能会有新旧版本 API 并存的情况,例如开放 Web API 给厂商串接,但同一个服务更新版本时,不一定所有厂商可以在同一时间都跟着更新他们的系统,但如果 ...

  9. core路由设置

    全局路由设置 app.UseMvc(routes => { routes.MapRoute( name: "areas", template: "{area:exi ...

  10. Vim 添加vimgdb支持

    ./configure --enable-gdb --prefix=/usr --enable-multibyte --enable-fontset --enable-xim --enable-gui ...