[题解] LuoguP4827 [国家集训队] Crash 的文明世界
这个题......我谔谔
首先可以考虑换根\(dp\),但到后来发现二项式定理展开过后需要维护\(k\)个值,同时每个值也要\(O(k)\)的时间按二项式定理算 当然fft优化过后就是k log k了...
这样复杂度是\(O(nk^2)\)的
当然\(FFT\)优化过后就变成\(O(nk \log k)\)了
这复杂度感觉是对的?但FFT的大常数好像被卡成\(50\)了 跟没优化一样,没试过qwq
然后发现又可以点分治,复杂度\(O(nk \log n)\),跑的好像比\(O(nk \log k)\)的FFT换根dp快?然后也收获了\(50\)分的好成绩...这真是令人谔谔
好像还有FFT优化点分治的,就可以过了,然而窝不会...
这里应该有一个更妙的做法。
注意到\(x^n\),其组合意义就是把\(n\)个不同的求随便往\(x\)个不同的盒子里扔的方案数。
这样有些盒子是空的,枚举有求的盒子个数,可以得到
\]
其实上面那个\(k\)的范围到\(n\)也是可以的,因为大于\(x\)过后下降幂将会变成\(0\),下面为了好看就到\(n\)吧
其中\(\begin{Bmatrix}n\\k\end{Bmatrix}\)表示第二类斯特林数,\([x]_k = \prod\limits_{i=0}^{k-1}(x-i)\),注意因为\(n \ge 1\),所以\(\begin{Bmatrix}n\\0\end{Bmatrix}=0\),\(k\)从\(0\)开始也没事。
更进一步的将下降幂写成组合数的形式有
\]
写到题目里的柿子去
\]
\]
\]
考虑怎么计算后面的\(\sum\),这个鉴于这棵树没有边权,以及组合数的递推性质,后面的东西可以换根\(dp\)。
具体的,先以\(1\)号点为根,令
\]
\(v \in x\)表示\(v\)在\(x\)的子树内,\(v \in son_x\)表示\(v\)是\(x\)的一个儿子
那么由于
\]
有转移
\]
特别的\(dp[x][0]=\sum\limits_{v \in son_x} dp[v][0]\)
那再令\(f[x][j]\)表示以\(x\)为根时的和,换根\(dp\)算出\(f[x][j]\)就好了,最后的答案
\]
预处理出斯特林数和阶乘就好了,复杂度\(O(nk + k^2)\)
\(Code:\)
#include <bits/stdc++.h>
using namespace std;
#define fore(i,x) for(int i=head[x],v=e[i].to;i;i=e[i].nxt,v=e[i].to)
const int N=5e4+10,P=10007;
inline int add(int x,int y){return (x+=y)>=P?x-P:x;}
inline int sub(int x,int y){return (x-=y)<0?x+P:x;}
inline int fpow(int x,int y)
{
int ret=1; for(;y;y>>=1,x=1ll*x*x%P)
if(y&1) ret=1ll*ret*x%P;
return ret;
}
int n,K;
struct edge
{
int to,nxt;
}e[N<<1];
int head[N],cnt=0;
inline void ade(int x,int y)
{e[++cnt]=(edge){y,head[x]};head[x]=cnt;}
inline void addedge(int x,int y){ade(x,y),ade(y,x);}
int dp[N][233],f[N][233];
void dfs(int x,int prev)
{
dp[x][0]=1;
fore(_,x) if(v!=prev)
{
dfs(v,x);
dp[x][0]=add(dp[x][0],dp[v][0]);
for(int i=1;i<=K;i++)
dp[x][i]=add(dp[x][i],add(dp[v][i-1],dp[v][i]));
}
}
void getans(int x,int prev)
{
fore(_,x) if(v!=prev)
{
static int fx[233];
fx[0]=sub(f[x][0],dp[v][0]);
f[v][0]=add(dp[v][0],fx[0]);
for(int i=1;i<=K;i++) fx[i]=sub(f[x][i],add(dp[v][i-1],dp[v][i]));
for(int i=1;i<=K;i++)
f[v][i]=add(dp[v][i],add(fx[i-1],fx[i]));
getans(v,x);
}
}
int S[233][233],fac[233];
int main()
{
scanf("%d%d",&n,&K);
fac[0]=1;for(int i=1;i<=K;i++) fac[i]=fac[i-1]*i%P;
S[0][0]=1;
for(int i=1;i<=K;i++)
for(int j=1;j<=K;j++)
S[i][j]=add(S[i-1][j]*j%P,S[i-1][j-1]);
for(int i=1,x,y;i<n;i++)
scanf("%d%d",&x,&y),addedge(x,y);
dfs(1,0);
for(int i=0;i<=K;i++) f[1][i]=dp[1][i];
getans(1,0);
for(int i=1;i<=n;i++)
{
int ans=0;
for(int j=0;j<=K;j++)
ans=add(ans,S[K][j]*fac[j]%P*f[i][j]%P);
printf("%d\n",ans);
}
return 0;
}
[题解] LuoguP4827 [国家集训队] Crash 的文明世界的更多相关文章
- [国家集训队] Crash 的文明世界(第二类斯特林数)
题目 [国家集训队] Crash 的文明世界 前置 斯特林数\(\Longrightarrow\)斯特林数及反演总结 做法 \[\begin{aligned} ans_x&=\sum\limi ...
- [国家集训队] Crash的文明世界
Description 给定一棵 \(n\) 个点的树,对于每个点 \(i\) 求 \(S(i)=\sum\limits_{j=1}^n \operatorname{dist(i,j)}^k\) .\ ...
- [国家集训队] Crash 的文明世界
不错的树形$ DP$的题 可为什么我自带大常数啊$ cry$ 链接:here 题意:给定一棵$ n$个节点的树,边权为$ 1$,对于每个点$ x$求$ \sum\limits_{i=1}^n dist ...
- 洛谷P4827 [国家集训队] Crash 的文明世界 [斯特林数,组合数,DP]
传送门 思路 又见到这个\(k\)次方啦!按照套路,我们将它搞成斯特林数: \[ ans_x=\sum_{i=0}^k i!S(k,i)\sum_y {dis(x,y) \choose i} \] 前 ...
- P4827 [国家集训队] Crash 的文明世界
传送门:洛谷 题目大意:设$$S(i)=\sum_{j=1}^ndis(i,j)^k$$,求$S(1),S(2),\ldots,S(n)$. 数据范围:$n\leq 50000,k\leq 150$ ...
- 解题:国家集训队 Crash 的文明世界
题面 这种套着高次幂的统计问题一般都要用到第二类斯特林数和自然数幂的关系:$a^k=\sum\limits_{i=0}^{k}S_k^iC_a^i*i!$ 那么对于每个点$x$有: $ans_x=\s ...
- 【[国家集训队] Crash 的文明世界】
先写一个五十分的思路吧 首先这道题有一个弱化版 [POI2008]STA-Station 相当于\(k=1\),于是就是一个非常简单的树形\(dp\)的\(up\ \ and\ \ down\)思想 ...
- P4827 [国家集训队] Crash 的文明世界(第二类斯特林数+树形dp)
传送门 对于点\(u\),所求为\[\sum_{i=1}^ndis(i,u)^k\] 把后面那堆东西化成第二类斯特林数,有\[\sum_{i=1}^n\sum_{j=0}^kS(k,j)\times ...
- 国家集训队 Crash 的文明世界(第二类斯特林数+换根dp)
题意 题目链接:https://www.luogu.org/problem/P4827 给定一棵 \(n\) 个节点的树和一个常数 \(k\) ,对于树上的每一个节点 \(i\) ,求出 \( ...
随机推荐
- 回文串[APIO2014](回文树)
题目描述 给你一个由小写拉丁字母组成的字符串 s.我们定义 s 的一个子串的存在值为这个子串在 s中出现的次数乘以这个子串的长度.对于给你的这个字符串 s,求所有回文子串中的最大存在值. 输入格式 一 ...
- VOC2012数据集提取自己需要的类的图片和对应的xml标签
根据需要修改路径和自己需要的类即可. import os import os.path import shutil fileDir_ann = r'/home/somnus/tttt/VOC2012/ ...
- Unity初识项目结构与面板
一.Unity的项目结构 Unity中的一个项目是有若干个场景组成的,每一个场景又是由若干个游戏对象组成的,每一个游戏对象身上有若干个组件,每一个组件有若干个属性. 项目——>场景——>游 ...
- dstat 监控命令详解
一.工具介绍 dstat的man手册对于该工具的解释: dstat - versatile tool for generating system resource statistics 系统资源多用途 ...
- vue中使用矢量图
1.打开矢量图库,将需要的图表添加至购物车 2.将购物车的图标添加到一个项目中(便于后期增加更新)并下载至本地 3.将这四个文件及iconfont.css添加至项目的assets中 4.打开iconf ...
- du与df的区别
我们知道,磁盘的整体数据是记录在superblock中的,但是每一个文件的容量信息则在inode当中记载的.因此,引出了两个查看这些数据信息的命令: df:列出文件系统的整体磁盘使用量.由于df主要读 ...
- python2学习------基础语法5(常用容器以及相关操作)
1.list(列表) #生成数据list a=[x for x in range(10)]; #print a; #遍历list for i in a: pass; #print i; #追加元素 a ...
- f_lseek
我在STM32中移植了fatfs文件系统,实现在SD卡对文件的读写.在普通读写中都没有问题,但是一旦我关闭文件系统,再次打开读写,之前写的数据就被覆盖.比如举个例子: u8 tx_buff ...
- iOS dismissViewControllerAnimated:completion:使用方法
我们都知道dismissViewControllerAnimated:completion:方法是针对被present出来的控制器的,一般我们这样使用:在一个控制器中present另外一个控制器A,然 ...
- GoJS实例2
复制如下内容保存到空白的.html文件中,用浏览器打开即可查看效果 <!DOCTYPE html> <html> <head> <meta name=&quo ...