【题目】H. Ember and Storm's Tree Game

【题意】Zsnuoの博客

【算法】动态规划+排列组合

【题解】题目本身其实并不难,但是大量干扰因素让题目显得很神秘。

参考:Zsnuoの博客

一、首先Ember必胜(考虑n个点连成一条链),故合法的树一定满足先手必胜。当Storm选择的链满足单调或单峰时,每一条链对答案贡献两对(i,op)

解释:单调时,考虑翻转最后一个数和从第二个数开始取负两种操作。单峰时,上凸考虑翻转顶峰和顶峰右侧的数,下凸考虑取负顶峰和顶峰右侧的数。

这样之后,我们就完全排除了博弈的因素(胜负和操作选择)。

二、问题转化为:令S表示满足 [ 包含n个点 ] 且 [ 每个点出度<=d ] 且 [ 所有树链编号单调或单峰 ] 的树,由(一)得最终答案ANS=2*n*(n-1)*|S|,即求|S|

其中单峰的路径比较容易考虑,S一定存在一个形态,满足所有单峰路径(u,v)的峰都是lca(u,v)。

证明:如果有一条路径峰不在LCA,那么lca(u,v)向上延伸一定是单调的,那选该峰作为根即可。

接下来考虑单调的路径(u,v)(u,v互不为祖先),易得如果lca(u,v)≠root就无法满足条件,所以不同单调性的指向根的链只能在根相遇。

也就是,每棵树都存在至少一个根root,满足所有以root为端点的路径都是单调的。接下来我们只须统计同一单调性的树的个数,在根处拼起来即可。

三、设f[i][j]表示包含i个点,根节点出度为j且满足 [ 所有节点编号均大于其父亲节点编号 ] 的树的个数。

同排列DP一样,我们关心的只有子树节点编号的大小关系,和具体是什么无关。所以我们每次都视为编号为1~i的节点来统计即可。

规定父亲节点编号为1,为了不重复统计,每次枚举根节点编号为2的子树大小k来转移,即:

$$f(i,j)=\sum _{k=1}^{i-1}f(i-k,j-1)*\binom{i-2}{k-1}*\sum _{l=0}^{d-1}f(k,l)$$

从未确定的i-2个点中选择一些来构成子树中的k-1个点,即C(i-2,k-1),然后再按大小关系当成2~k排列。

令$sum(i)=\sum _{j=0}^{d-1}f(i,j)$,就可以前缀和优化:

$$f(i,j)=\sum _{k=1}^{i-1}f(i-k,j-1)*\binom{i-2}{k-1}*sum(k)$$

预处理组合数,初始化f(1,0)=1。(这样就不用理会非素数模数了,不然又得搞CRT那套,太麻烦了……)

四、在根节点处拼接。

假设两个根节点都为root的不同单调性的树拼接(根节点重合),枚举第一单调性的点数i,第一单调性所占度数j,第二单调性所占度数k,即:

$$|S|=\sum _{i=1}^{n}\sum _{j+k\leq d}f(i,j)*f(n-i+1,k)$$

最后一个问题,如果一棵树S存在多个root(例如n=2),那么这多个root一定构成一条单调链,其中一端是j=1且k≠1,另一端是j≠1且k=1,中间是j=1且k=1,我们只在第一种情况统计这棵树。

那么,最终:

$$|S|=\sum _{i=1}^{n}\sum _{j+k\leq d,k\neq 1}f(i,j)*f(n-i+1,k)$$

复杂度O(n^3)。

#include<cstdio>
const int maxn=; long long n,d,MOD,c[maxn][maxn],sum[maxn],ans,f[maxn][maxn];
int main(){
scanf("%lld%lld%lld",&n,&d,&MOD);
for(int i=;i<=n;i++){
c[i][]=;
for(int j=;j<=i;j++)c[i][j]=(c[i-][j-]+c[i-][j])%MOD;
}
f[][]=sum[]=;
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-]*c[i-][k-]%MOD*sum[k])%MOD;
}
if(j!=d)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;
printf("%lld",*n*(n-)*ans%MOD);
return ;
}

【CodeForces】914 H. Ember and Storm's Tree Game 动态规划+排列组合的更多相关文章

  1. 【codeforces 914H】Ember and Storm's Tree Game

    原题链接 Description Ember和Storm正在玩游戏.首先,Ember构造一棵n个节点且每个节点度数不超过d的带节点编号的树T.然后,Storm选择两个不同的节点u和v,并写下从u到v路 ...

  2. Codeforces 914H Ember and Storm's Tree Game 【DP】*

    Codeforces 914H Ember and Storm's Tree Game 题目链接 ORZ佬 果然出了一套自闭题 这题让你算出第一个人有必胜策略的方案数 然后我们就发现必胜的条件就是树上 ...

  3. Codeforces Round #309 (Div. 2) C. Kyoya and Colored Balls 排列组合

    C. Kyoya and Colored Balls Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/contes ...

  4. Codeforces Round #167 (Div. 2) D. Dima and Two Sequences 排列组合

    题目链接: http://codeforces.com/problemset/problem/272/D D. Dima and Two Sequences time limit per test2 ...

  5. Codeforces 840C. On the Bench 动态规划 排列组合

    原文链接https://www.cnblogs.com/zhouzhendong/p/CF840C.html 题解 首先,我们可以发现,如果把每一个数的平方因子都除掉,那么剩下的数,不相等的数都可以相 ...

  6. Codeforces Gym 100187D D. Holidays 排列组合

    D. Holidays Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/gym/100187/problem/D ...

  7. 【CodeForces】889 C. Maximum Element 排列组合+动态规划

    [题目]C. Maximum Element [题意]给定n和k,定义一个排列是好的当且仅当存在一个位置i,满足对于所有的j=[1,i-1]&&[i+1,i+k]有a[i]>a[ ...

  8. Codeforces 991E. Bus Number (DFS+排列组合)

    解题思路 将每个数字出现的次数存在一个数组num[]中(与顺序无关). 将出现过的数字i从1到num[i]遍历.(i from 0 to 9) 得到要使用的数字次数数组a[]. 对于每一种a使用排列组 ...

  9. [Codeforces 1228E]Another Filling the Grid (排列组合+容斥原理)

    [Codeforces 1228E]Another Filling the Grid (排列组合+容斥原理) 题面 一个\(n \times n\)的格子,每个格子里可以填\([1,k]\)内的整数. ...

随机推荐

  1. C++ Primer Plus学习:第三章

    C++入门第三章:处理数据 面向对象编程(OOP)的本质是设计并扩展自己的数据类型. 内置的C++数据类型分为基本类型和复合类型. 基本类型分为整数和浮点数. 复合类型分为数组.字符串.指针和结构. ...

  2. Tomcat安装及配置详解

    Tomcat安装及配置详解   一,Tomcat简介 Tomcat 服务器是一个免费的开放源代码的Web 应用服务器,Tomcat是Apache 软件基金会(Apache Software Found ...

  3. delphi使用SQL的教程4(使用Params属性为参数赋值 )

    17.4.1 使用Params属性为参数赋值   TQuery部件具有一个Params属性,它们在设计时不可用,在程序运行过程中可用,并且是动态建立的,当为TQuery部件编写动态SQL 语句时, D ...

  4. BZOJ 2141 排队(树状数组套主席树)

    解法很多的题,可以块套树状数组,可以线段树套平衡树.我用的是树状数组套主席树. 题意:给出一段数列,m次操作,每次操作是交换两个位置的数,求每次操作后的逆序对数.(n,m<=2e4). 对于没有 ...

  5. 辣鸡蒟蒻Klaier的一些计划

    需要熟练的东西:cdq分治,堆,树链剖分,tarjan及其它一些图论算法,网络流,kmp,字符串哈希,线段树主席树,树状数组,斜率优化dp 需要学的东西:lct,后缀数组,AC自动机,平衡树 球队收益 ...

  6. [Code Festival 2017 qual A] B: flip

    题意 给出一个n行m列初始全白的矩阵,每次可以翻转一行/一列的全部格子的颜色.问任意次操作后能否使得恰好有k个黑色格子. n,m<=1000 分析 显然要么翻转一次要么不翻转. 最终黑色格子数只 ...

  7. wp开发(三)--赚取收益篇

    App开发完毕了,是否有赚取收益的想法呢?下面很浅显地介绍两种常用赚取收益的方法. 一. 收费 在发布应用时,可以对应用进行定价,发布到商城之后,用户付费才可以下载,当然也可以提供试用版.收益状况可以 ...

  8. windows上配置连接git

    Windows7上安装Git   我首先是百度到了这个网站:https://git-scm.com/download/win 这个网站上有下载链接,你可以根据你的系统选择不同的下载链接,我的是Win7 ...

  9. vdbench测试过程中遇到的小问题

    1.报Slave hd2-0 prematurely terminated 错误 首先根据提示查看hd2-0.stdout.html文件获取更多的错误信息,这个问题一般是未安装vdbench或者路径不 ...

  10. URL补充

    1. 笔记 2. 关于默认值的解释:在url里面,可以直接给views.index传递一个默认值. index函数接收一个形式参数. 在urls.py中,可以直接传递一个实参(也就是默认值). 打印结 ...