[CSP-S模拟测试]:超级树(DP)
题目传送门(内部题5)
输入格式
一行两个整数$k$、$mod$,意义见上。
输出格式
一行一个整数,代表答案。
样例
样例输入1:
2 100
样例输出1:
9
样例输入2:
3 1000
样例输出2:
245
样例输入3:
20 998244353
样例输出3:
450500168
数据范围与提示
样例解释:
对于第一组样例,将节点如图编号,共有9条不同的路径:1,2,3,1-2,2-1,1-3,3-1,2-1-3,3-1-2。
限制与约定:
对于$10\%$的数据,$k \leqslant 4$。
对于$40\%$的数据,$k \leqslant 10$。
对于$60\%$的数据,$k \leqslant 100$。
另有$10\%$的数据,$mod=998244353$。
对于所有数据,$1 \leqslant k \leqslant 300,1 \leqslant mod \leqslant {10}^9$。
题解
有谁能想到$DP$?举个爪
显然k-超级树是由两个$(k-1)-$超级树合在一起加了个根组成的。
那么好吧,确定了是个$DP$,但是又有谁能想到$DP$的意义呢?
定义$dp[i][j]$表示一棵$i-$超级树,有$j$条没有公共点的路径的方案数。
好像说的有点乱,那么我们拿样例来唠两句:
$dp[2][1]=9$,这很显然,就是样例解释中那$9$种方案。
$dp[2][2]=7$,这七种方案分别是$(1,2)$,$(1,3)$,$(2,3)$,$(2,1-3)$,$(2,3-1)$,$(3,1-2)$,$(3,2-1)$。
$dp[2][3]=1$,这种方案是$(1,2,3)$。
$dp[2][4]=0$,因为一共只有$3$个点,所以我们找不到方案了。
现在开始来推式子吧~
我们来考虑$dp[i]$对$dp[i+1]$的贡献:枚举左子树的路径条数l和有子树的路径条数$r$,记$num=dp[i][l] \times dp[i][r]$。
转移分一下五种情况:
$1.$什么也不做:$dp[i+1][l+r]+=num$。
$2.$跟自己作为一条新路径:$dp[i+1][l+r+1]+=num$,不要忘了根本身还有一条。
$3.$根连接到左子树(或右子树)的某条路径上:$dp[i+1][l+r]+=2 \times num \times (l+r)$。
对于$3$的理解:从左子树里选一条边,延长其终点,连接根的路径条数为$sum \times l$;同理,将起点做如上操作,路径条数也为$sum \times l$;再同理,右子树也是这种操作,路径条数为$2 \times sum \times r$;总的路径条数即为:$dp[i+1][l+r]+=2 \times num \times (l+r)$。
$4.$根连接左子树和右子树的各一条路径:$dp[i+1][l+r-1]+=2 \times num \times l \times r$。
对于$4$的理解:从左子树中选一条边,延长其终点,经根节点连向有子树中一条边的起点,路径条数为:$sum \times l \times r$;同理,从右子树中选一条边也是如此,总的路径条数为:$dp[i+1][l+r-1]+=2 \times num \times l \times r$。
$5.$根连接左子树(或右子树)的两条路径:$dp[i+1][l+r-1]+=num \times (l \times (l-1)+r \times (r-1))$。
对于$5$的理解:左子树中选一条边的终点连向根节点,再将这条边继续延长,连向做字数中另外一条边的起点,路径条数为:$num \times (l \times (l-1))$;右子树中也是如此,路径条数为:$num \times (r \times (r-1))$;总的路径条数即为:$dp[i+1][l+r-1]+=num \times (l \times (l-1)+r \times (r-1))$。
边界为$dp[1][0]=dp[1][1]=1$,答案为$dp[k][1]$。
看起来第二维状态可能有$2^k$那么大,但注意到从$dp[i]$转移到$dp[i+1]$时,路径的条数最多减少1条,因此第二维只有$k$个状态对最终的状态有影响,只$dp$这些状态即可。
时间复杂度$O(k^3)$。
代码时刻
#include<bits/stdc++.h>
using namespace std;
int k;
long long mod,num;
long long dp[310][310];
int main()
{
dp[1][0]=dp[1][1]=1;
scanf("%d%lld",&k,&mod);
for(int i=1;i<k;i++)
for(int l=0;l<=k-i+1;l++)
for(int r=0;r<=k-i-l+2;r++)
{
num=dp[i][l]*dp[i][r]%mod;
dp[i+1][l+r]=(dp[i+1][l+r]+num)%mod;//情况1
dp[i+1][l+r+1]=(dp[i+1][l+r+1]+num)%mod;//情况2
dp[i+1][l+r]=(dp[i+1][l+r]+2*num*(l+r))%mod;//情况3
dp[i+1][l+r-1]=(dp[i+1][l+r-1]+2*num*l*r)%mod;//情况4
dp[i+1][l+r-1]=(dp[i+1][l+r-1]+num*(l*(l-1)+r*(r-1)))%mod;//情况5
}
printf("%lld",dp[k][1]%mod);
return 0;
}
rp++
[CSP-S模拟测试]:超级树(DP)的更多相关文章
- 【NOIP模拟赛】超级树 DP
这个题我在考试的时候把所有的转移都想全了就是新加一个点时有I.不作为II.自己呆着III.连一个IV.连接两个子树中的两个V连接一个子树中的两个,然而V我并不会转移........ 这个题的正解体现了 ...
- [CSP-S模拟测试]:F(DP+线段树)
题目传送门(内部题49) 输入格式 第一行四个整数$n,q,a,b$.接下来$n$行每行一个整数$p_i$. 输出格式 一行一个整数表示答案. 样例 样例输入: 10 3 3 7 样例输出: 数据范围 ...
- [CSP-S模拟测试]:最小值(DP+乱搞)
题目背景 $Maxtir$更喜欢序列的最小值. 题目传送门(内部题128) 输入格式 第一行输入一个正整数$n$和四个整数$A,B,C,D$. 第二行输入$n$个整数,第$i$个数表示$a_i$. 输 ...
- [CSP-S模拟测试]:题(DP)
题目描述 由于出题人赶时间所以没办法编故事来作为背景.一开始有$n$个苹果,$m$个人依次来吃苹果,第$i$个人会尝试吃$u_i$或$v_i$号苹果,具体来说分三种情况.$\bullet 1.$两个苹 ...
- [CSP-S模拟测试]:tree(DP)
题目传送门(内部题57) 输入格式 第一行包含一个数:$n$表示树的节点数.接下来$n-1$行,每行包含两个数:$u,v$表示无根树的一条边. 输出格式 输出$n$行,第$i$行包含一个浮点数,保留三 ...
- [CSP-S模拟测试]:biology(DP)
题目传送门(内部题23) 输入格式 第一行有$2$个整数$n,m$.接下来有$n$行,每行$m$个整数,表示$a$数组.接下来有$n$行,每行$m$个整数,表示$b$数组. 输出格式 一行一个整数表示 ...
- HZOI2019 超级树 dp
题面:https://www.cnblogs.com/Juve/articles/11207540.html(密码)————————————————>>> 题解: 官方题解: 考虑d ...
- [CSP-S模拟测试]:B(DP+数学)
题目传送门(内部题45) 输入格式 第一行$3$个整数$n,m,P$.第二行$m$个整数,表示$m$次询问. 输出格式 一行$m$个整数表示答案. 样例 样例输入1: 2 4 40 1 2 3 样例输 ...
- [CSP-S模拟测试]:蛇(DP+构造+哈希)
题目传送门(内部题140) 输入格式 前两行有两个长度相同的字符串,描述林先森花园上的字母. 第三行一个字符串$S$. 输出格式 输出一行一个整数,表示有多少种可能的蛇,对$10^9+7$取模. 样例 ...
随机推荐
- IIS7下配置web.config隐藏index.php
<?xml version="1.0" encoding="UTF-8"?> <configuration> <system.we ...
- kafka消费者脚本无法启动问题
console-consumer can't rebalance after 4 retries 解决方案:kafka0.9版本换成1.0版本 究竟是怎么回事我也不知道
- Keil共存的方法 - Keil MDK兼容Keil C51,实操可行
记录一下成功使Keil MDK和Keil C51共存的过程! 之前一直用Keil C51开发,最近需要用到ARM9内核的IC,就需要Keil C51和Keil MDK共存.看了一下网上几个教程,方法大 ...
- [Web 前端] 025 js 的对象、数组和数学对象
1. Javascript 对象 1.1 创建对象 1.1.1 使用原始的方式创建内置对象 var myObject = new Object(); myObject.name = "lij ...
- word定义多级列表
1.单击开始选项卡里的多级列表按钮,在下拉列表中选择定义新的多级列表 2.先设置第一级编号,选择阿拉伯数字1,2,3,...,并在自动编号“1”的左右分别输入“第”“章”,级别链接到样式选择标题一 3 ...
- oracle数据库中的存储函数
oracle中的存储函数,和系统内的函数类似,可以像调用系统函数一样调用存储函数.它与存储过程的唯一区别就是存储过程没有return返回值,存储函数可以与存储过程互换,存储函数可以在存储过程中调用. ...
- 使用css3的repeating-linear-gradient画虚线
还在用 border-style: dashed 画虚线吗?虽然也是虚线,但是不能控制每一个虚线的宽度 .dashed { height: 1px; background-image: repeati ...
- 以当前时间作为GUID的方法
在C#中,系统提供了GUID类,用户可以通过该类来获得128位的唯一标识,但是该标识不具有可读性,很难把该GUID显示在界面上,以当前时间精确到毫秒来作为GUID是一个比较不错的做法,但是DateTi ...
- JavaEE高级-MyBatis学习笔记
一.MyBatis简介 - MyBatis 是支持定制化 SQL.存储过程以及高级映射的优秀的持久层框架. - MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集. - My ...
- 025-Cinder服务-->安装并配置一个本地存储节点(ISCSI)
一:Cinder提供块级别的存储服务,块存储提供一个基础设施为了管理卷,以及和OpenStack计算服务交互,为实例提供卷.此服务也会激活管理卷的快照和卷类型的功能,块存储服务通常包含下列组件:cin ...