博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
CF735E Ostap and Tree
阅读量:7286 次
发布时间:2019-06-30

本文共 1583 字,大约阅读时间需要 5 分钟。

比较毒瘤的树形DP,子状态难想。这是主要是搬运一篇题解。

\(f[i][j]\)表示\(i\)的子树中离\(i\)最近黑点的距离为\(j\),且距离超过\(j\)的点都被满足的方案数。转移时新建一个临时数组\(tmp\)保存转移后的\(f[x]\)。设\(y\)\(x\)的子结点,枚举\(f[x][i]\)\(f[y][j]\),转移如下:

  1. \(i+j≤2k\),则此时\(min(i,j+1)≤k\),对于长度为\(i+j+1\)的链上的所有点都可以找到一边距离\(≤k\),因此状态合并以后是合法状态,转移\(tmp[min(i,j+1)]+=f[x][i]×f[y][j]\)

  2. \(i+j>2k\),则此时\(max(i,j+1)>k\),链上肯定会存在一些点两边都够不到,转移\(tmp[max(i,j+1)]+=f[x][i]×f[y][j]\)

初始状态\(f[x][0]=1\),表示不考虑子树内的情况,选择自己的方案数为\(1\)\(f[x][k+1]=1\),表示自己本身不满足,但子结点都被满足的情况,主要是方便转移。

答案为\(∑i<=kf[root][i]\)

时间复杂度\(O(nk2)\)

代码如下

#include
#include
#include
#include
typedef long long int64;inline int getint() { register char ch; while(!isdigit(ch=getchar())); register int x=ch^'0'; while(isdigit(ch=getchar())) x=(((x<<2)+x)<<1)+(ch^'0'); return x;}const int N=101,K=41,mod=1e9+7;int k,f[N][K],tmp[K];std::forward_list
e[N];inline void add_edge(const int &u,const int &v) { e[u].push_front(v); e[v].push_front(u);}void dfs(const int &x,const int &par) { f[x][0]=f[x][k+1]=1; for(int &y:e[x]) { if(y==par) continue; dfs(y,x); std::fill(&tmp[0],&tmp[k*2]+1,0); for(register int i=0;i<=k*2;i++) { for(register int j=0;j<=k*2;j++) { (tmp[i+j<=k*2?std::min(i,j+1):std::max(i,j+1)]+=(int64)f[x][i]*f[y][j]%mod)%=mod; } } std::copy(&tmp[0],&tmp[k*2]+1,f[x]); }}int main() { const int n=getint();k=getint(); for(register int i=1;i

转载于:https://www.cnblogs.com/ilverene/p/10339467.html

你可能感兴趣的文章
JavaScript系列:ECMAScript原始类型
查看>>
centos反编译APK包
查看>>
CSS系列:CSS中盒子的浮动与定位
查看>>
windows 用户用户组迁移
查看>>
Linux系统扩充2
查看>>
linux新手的心得
查看>>
我的友情链接
查看>>
zabbix表字段类型和value type问题
查看>>
shoususaiBti
查看>>
solr5.5.5独立部署(不使用tomcat)
查看>>
WINDOWSXP启动时直接进入系统而无需入用户名和密码
查看>>
论测试的主要责任
查看>>
关于测试团队的组织
查看>>
如何解决WEB性能测试中的验证码问题
查看>>
WinPe3.1启动系统逐步完善专题02:软件环境搭建
查看>>
思科模拟器——使用路由器分割局域网
查看>>
Tomcat日志配置
查看>>
Apache Spark源码走读之14 -- Graphx实现剖析
查看>>
2017年以后武汉的房价还会涨吗?
查看>>
10个免费开源的JS音乐播放器插件
查看>>