【codeforces 914H】Ember and Storm's Tree Game
Description
Ember和Storm正在玩游戏。首先,Ember构造一棵n个节点且每个节点度数不超过d的带节点编号的树T。然后,Storm选择两个不同的节点u和v,并写下从u到v路径上的节点编号,记为序列 a1, a2... ak 。最后,Ember在序列中选择一个位置 i(1 ≤ i < k),并在以下两个操作选择一个执行:
- 翻转 ai+1... ak 并将这一段加上ai,操作后序列变为 a1, ... ai, ak + ai, ak-1 + ai, ... ai+1 + ai
- 取负 ai+1... ak 并将这一段加上ai,操作后序列变为 a1, ... ai, - ai+1 + ai, - ai+2 + ai, ... - ak + ai
如果最后的序列是严格单调的,则Ember获胜,否则Storm获胜。
游戏情形可以用一个元组 (T, u, v, i, op) 来描述,op为翻转或是取负取决于Ember的决策。若Ember和Storm都使用最优策略(若有多种必胜策略,任选一种执行;若必败,也任选一种执行),试统计所有可能的游戏情形的数量,并输出其取模m的结果。
Input
仅一行,给出 n,d,m。 (2 ≤ n ≤ 200, 1 ≤ d < n, 1 ≤ m ≤ 2·109).
Output
输出一个数字——所有可能的游戏情形的数量取模m之后的结果。
首先,Ember一定会构造出一棵能让自己必胜的树。而Ember获胜当而仅当原序列$a$为单调的或是单峰的;且对于每一个合法的序列,有2种合法的$(i,op)$的组合。没有什么好证明的……在草稿纸上自己模拟一下两种操作就可以得到了。
问题转换为:统计满足以下条件的树的数量$S$:1. 包含$n$个节点,2. 每个节点度数不超过$d$,3. 树上任意两个节点间路径的编号序列为单调的或单峰的。最终答案为 $2\cdot n\cdot(n-1)\cdot S$ 。
而对于一棵合法的树,一定存在一个特殊点,满足以这个节点为起点或终点的所有路径都是单调的。为了方便统计,我们令合法树的根节点为特殊点。观察可得,对于一棵合法树,除根节点以外的子树都满足:父亲节点编号大于儿子编号,或是父亲编号小于儿子编号。所以我们只需要统计这两种情况的答案,然后在根节点处拼起来即可。而实际上,这两种情况是等价的。
令$f(i,j)$表示节点数为$i$,根节点度数为$j$,且父亲编号小于儿子编号的方案数。
枚举当前要拼接的子树大小$k$,钦定根节点编号最小,拼接过来的子树的根节点编号次小,可得到以下递推公式:
$$f(i,j)=\sum _{k=1}^{i-1}f(i-k,j-1)\cdot \binom{i-2}{k-1}\cdot \sum _{l=1}^{d-1}f(k,l)$$
令 $sum(i)=\sum _{j=1}^{d-1}f(i,j)$,可得:
$$f(i,j)=\sum _{k=1}^{i-1}f(i-k,j-1)\cdot \binom{i-2}{k-1}\cdot sum(k)$$
时间复杂度为 $O(n^{3})$ ,初始化 $f(1,0)=sum(1)=1$ 。
(这种方法是在评论区看到的……然后参考了一下wxh大爷的博客。官方题解给了另一种统计f数组的方式,要稍微复杂一些,以及因为不保证m是质数,会有一些细节需要处理。详见官方题解,细节处理详见评论区。)
统计出$f$数组后就可以开始拼接了,枚举满足父亲节点编号小于儿子编号的点数$i$、度数$j$, 满足父亲节点编号大于儿子编号的度数$k$,可得到以下公式:
$$S=\sum _{i=0}^{n-1}\sum _{j=0}^{d}\sum _{k=0}^{d-j}f(i+1,j)\cdot f(n-i,k)$$
而实际上一棵合法树是可以有多个合法根的,比如最简单的$n=2$的情况,合法根既可以是$1$也可以是$2$。我们可以得出另一个结论,如果一棵树有多个合法根,那么这些点一定构成一条单调链,一端是$j=1$且$k≠1$,另一端是$j≠1$且$k=1$,中间是$j=1$且$k=1$,我们把这棵树放在第一种情况统计。
得到最终公式:
$$S=\sum _{i=0}^{n-1}\sum _{j+k\leq d,k\neq 1}f(i+1,j)\cdot f(n-i,k)$$
代码如下:
#include<cstdio>
#include<algorithm>
#include<cstring>
#define LL long long
using namespace std;
const int N=;
int n,d,mod;
LL ans,sum[N],c[N][N],f[N][N];
int main()
{
scanf("%d%d%d",&n,&d,&mod);
for(int i=;i<=n;i++)c[i][]=;
for(int i=;i<=n;i++)
for(int j=;j<=i;j++)
c[i][j]=(c[i-][j]+c[i-][j-])%mod;
sum[]=;f[][]=;
for(int i=;i<=n;i++)
{
for(int j=;j<=d;j++)
for(int k=;k<i;k++)
f[i][j]=(f[i][j]+f[i-k][j-]*sum[k]%mod*c[i-][k-]%mod)%mod;
for(int j=;j<=d-;j++)
sum[i]=(sum[i]+f[i][j])%mod;
}
for(int i=;i<=n-;i++)
for(int j=;j<=d;j++)
for(int k=;j+k<=d;k++)
if(k!=)ans=(ans+f[i+][j]*f[n-i][k]%mod)%mod;
printf("%lld",*n*(n-)*ans%mod);
return ;
}
【codeforces 914H】Ember and Storm's Tree Game的更多相关文章
- Codeforces 914H Ember and Storm's Tree Game 【DP】*
Codeforces 914H Ember and Storm's Tree Game 题目链接 ORZ佬 果然出了一套自闭题 这题让你算出第一个人有必胜策略的方案数 然后我们就发现必胜的条件就是树上 ...
- 【CodeForces】914 H. Ember and Storm's Tree Game 动态规划+排列组合
[题目]H. Ember and Storm's Tree Game [题意]Zsnuoの博客 [算法]动态规划+排列组合 [题解]题目本身其实并不难,但是大量干扰因素让题目显得很神秘. 参考:Zsn ...
- 【codeforces 415D】Mashmokh and ACM(普通dp)
[codeforces 415D]Mashmokh and ACM 题意:美丽数列定义:对于数列中的每一个i都满足:arr[i+1]%arr[i]==0 输入n,k(1<=n,k<=200 ...
- 【codeforces 707E】Garlands
[题目链接]:http://codeforces.com/contest/707/problem/E [题意] 给你一个n*m的方阵; 里面有k个联通块; 这k个联通块,每个连通块里面都是灯; 给你q ...
- 【codeforces 707C】Pythagorean Triples
[题目链接]:http://codeforces.com/contest/707/problem/C [题意] 给你一个数字n; 问你这个数字是不是某个三角形的一条边; 如果是让你输出另外两条边的大小 ...
- 【codeforces 709D】Recover the String
[题目链接]:http://codeforces.com/problemset/problem/709/D [题意] 给你一个序列; 给出01子列和10子列和00子列以及11子列的个数; 然后让你输出 ...
- 【codeforces 709B】Checkpoints
[题目链接]:http://codeforces.com/contest/709/problem/B [题意] 让你从起点开始走过n-1个点(至少n-1个) 问你最少走多远; [题解] 肯定不多走啊; ...
- 【codeforces 709C】Letters Cyclic Shift
[题目链接]:http://codeforces.com/contest/709/problem/C [题意] 让你改变一个字符串的子集(连续的一段); ->这一段的每个字符的字母都变成之前的一 ...
- 【Codeforces 429D】 Tricky Function
[题目链接] http://codeforces.com/problemset/problem/429/D [算法] 令Si = A1 + A2 + ... + Ai(A的前缀和) 则g(i,j) = ...
随机推荐
- CSS--字体|垂直居中|background
一,字体的设置 二,垂直居中 2.1,单行文本垂直居中 2.2,多行文本垂直居中 2.3,绝对定位元素垂直居中 三.颜色的表示法 四.background ---------------------- ...
- kernel笔记——VFS
vfs(the virtual filesystem, virtual file switch)为应用程序访问文件提供了统一的接口,如read.write.open等. 下面我们看加载文件系统模块.格 ...
- 前端面试回顾---javascript的面向对象
转:https://segmentfault.com/a/1190000011061136 前言 前一阵面试,过程中发现问到一些很基础的问题时候,自己并不能很流畅的回答出来.或者遇到一些基础知识的应用 ...
- day2 and day3 总结-python基础-坚持就是胜利
今日份快捷键学习,pycharm中按CTRL+ALT+L 自动规范化操作,不过和QQ的快捷键会有冲突,建议更改QQ的 知识点: 1.编码 2.while循环 3.运算符 4.数字int 5.布尔值 6 ...
- 三 Struts2 添加返回数据
一.struts2如何获取servletAPI1.解耦和的方式获取封装后的Map对象,这种获取的map对象只有存取数据的功能. Map request = (Map) ActionContext.ge ...
- esp8266 免费wifi强推广告神器(4) 发现当前WIFI下的用户数目,IP,MAC请求http信息 在用户请求跳转后跳转
需求: 1 获取当前连接客户端的HTTP请求各种信息 方法 get http 请求路径 例如 /index.html / /pic.jpg 请求版本 HTTP/1.0 HT ...
- AI adanet
adanet是一个基于Tensorflow的轻量级框架,只需要很少的专业干预,就可以自动学习出高质量的模型.在最近的AutoML成果上构建,既快速又灵活,还可以保证学习质量. adanet提供通用框架 ...
- SQL Server中NULL的一个测试
我们都知道SQL Server中NULL是一个很特殊的存在,因为NULL不会等于任何值,且NULL也不会不等于任何值.对于NULL我们只能使用IS或IS NOT关键字来进行比较. 我们先来看看下面一个 ...
- 【北航软件工程】Alpha阶段前端页面编写及服务器部署
前端页面编写 虽然之前对html语法有过一些了解,但是完全没有编写前端页面的经验,和我合作的czy大概也是这么个情况.在Alpha阶段的前端页面编写过程中,我们是摸着石头过河,html是个入门很快专精 ...
- Python的dnspython库使用指南
因为平时在测试DNS的时候有些操作手动完成不方便,所以需要用到脚本,而在Python里dnspython这个用于DNS操作的库十分强大,但是无奈网上大部分资料只列举了少部分的用法,所以记录一下我平时使 ...