Luogu4099 HEOI2013 SAO 组合、树形DP

时间:2021-09-01 01:41:09

传送门


值得注意的是一般的DAG的拓扑序列数量是NP问题,所以不能直接入手

题目中给出的图可以看做是一个树形图,虽然方向比较迷。考虑使用树形图的性质

不妨任选一个点为根做树形DP,注意到数的位置与方案数相关,所以也要设在状态内。故设\(f_{i,j}\)表示对于\(i\)及\(i\)的子树所有点构成的拓扑序列,\(i\)排在第\(j\)位的方案数,通过一个儿子一个儿子地合并来转移。

对于当前计算的点\(u\)的某一个儿子\(v\)已经算完,正要和\(u\)合并。设\(sz_u\)表示\(u\)和\(u\)的已合并子树的点数之和,并设\(u < v\),那么序列中\(v\)要在\(u\)的后面。

先枚举\(u\)在已合并序列中的位置\(j\),然后枚举\(v\)的子树对应的拓扑序列中有\(k\)个放在\(u\)的前面,不难得到转移方程:\(dp_{u,j+k} \leftarrow dp_{u,j} \times C_{j-1+k}^k \times C_{sz_u + sz_v - j - k} ^ {sz_v - k} \times \sum\limits_{i=k+1}^{sz_v} dp_{v,k}\)

如果\(u>v\)也是差不多的

前后缀和优化最后的\(\sum dp_{v,k}\)并预处理组合数

关于复杂度,注意到每一次合并的复杂度为"已合并的儿子大小"\(\times\)"当前合并的儿子大小",可以看作任意两个点只会在它们的LCA处产生\(1\)的复杂度,所以总复杂度是\(O(n^2)\)的。

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<ctime>
#include<cctype>
#include<algorithm>
#include<cstring>
#include<iomanip>
#include<queue>
#include<map>
#include<set>
#include<bitset>
#include<stack>
#include<vector>
#include<cmath>
#include<cassert>
//This code is written by Itst
using namespace std; inline int read(){
int a = 0;
char c = getchar();
bool f = 0;
while(!isdigit(c) && c != EOF){
if(c == '-')
f = 1;
c = getchar();
}
if(c == EOF)
exit(0);
while(isdigit(c)){
a = a * 10 + c - 48;
c = getchar();
}
return f ? -a : a;
} const int MOD = 1e9 + 7;
int N , cntEd;
struct Edge{
int end , upEd;
}Ed[2003];
int head[1007] , C[1007][1007] , dp[1007][1007] , sz[1007] , pre[1007][1007] , suf[1007][1007] , tmp[1007]; inline void addEd(int a , int b){
Ed[++cntEd].end = b;
Ed[cntEd].upEd = head[a];
head[a] = cntEd;
} inline char getc(){
char c = getchar();
while(c == ' ' || c == '\n' || c == '\r')
c = getchar();
return c;
} void init(){
C[0][0] = 1;
for(int i = 1 ; i <= 1000 ; ++i){
C[i][0] = 1;
for(int j = 1 ; j <= 1000 ; ++j)
C[i][j] = (C[i - 1][j] + C[i - 1][j - 1]) % MOD;
}
} void dfs(int u , int p){
sz[u] = 1;
dp[u][1] = 1;
for(int i = head[u] ; i ; i = Ed[i].upEd)
if(Ed[i].end != p){
dfs(Ed[i].end , u);
int v = Ed[i].end;
sz[u] += sz[v];
memset(tmp , 0 , sizeof(tmp));
if(i & 1)
for(int j = sz[u] - sz[v] ; j ; --j)
for(int k = 0 ; k < sz[v] ; ++k)
tmp[j + k] = (tmp[j + k] + 1ll * dp[u][j] * suf[v][k + 1] % MOD * C[j + k - 1][k] % MOD * C[sz[u] - j - k][sz[v] - k]) % MOD;
else
for(int j = sz[u] - sz[v] ; j ; --j)
for(int k = 1 ; k <= sz[v] ; ++k)
tmp[j + k] = (tmp[j + k] + 1ll * dp[u][j] * pre[v][k] % MOD * C[j + k - 1][k] % MOD * C[sz[u] - j - k][sz[v] - k]) % MOD;
memcpy(dp[u] , tmp , sizeof(tmp));
}
for(int i = 1 ; i <= sz[u] ; ++i)
pre[u][i] = (pre[u][i - 1] + dp[u][i]) % MOD;
for(int i = sz[u] ; i ; --i)
suf[u][i] = (suf[u][i + 1] + dp[u][i]) % MOD;
} int main(){
#ifndef ONLINE_JUDGE
freopen("in","r",stdin);
//freopen("out","w",stdout);
#endif
init();
for(int T = read() ; T ; --T){
N = read();
for(int i = 1 ; i < N ; ++i){
int a = read() + 1;
char c = getc();
int b = read() + 1;
if(c == '>')
swap(a , b);
addEd(a , b);
addEd(b , a);
}
dfs(1 , 0);
cout << pre[1][N] << endl;
memset(dp , 0 , sizeof(dp));
memset(pre , 0 , sizeof(pre));
memset(suf , 0 , sizeof(suf));
memset(head , 0 , sizeof(head));
cntEd = 0;
}
return 0;
}
=

Luogu4099 HEOI2013 SAO 组合、树形DP的更多相关文章

  1. 洛谷P4099 &lbrack;HEOI2013&rsqb;SAO(树形dp)

    传送门 HEOI的题好珂怕啊(各种意义上) 然后考虑树形dp,以大于为例 设$f[i][j]$表示$i$这个节点在子树中排名第$j$位时的总方案数(因为实际只与相对大小有关,与实际数值无关) 我们考虑 ...

  2. P4099 &lbrack;HEOI2013&rsqb;SAO(树形dp)

    P4099 [HEOI2013]SAO 我们设$f[u][k]$表示以拓扑序编号为$k$的点$u$,以$u$为根的子树中的元素所组成的序列方案数 蓝后我们在找一个以$v$为根的子树. 我们的任务就是在 ...

  3. 洛谷 P4099 - &lbrack;HEOI2013&rsqb;SAO(树形 dp)

    题面传送门 题意: 有一个有向图 \(G\),其基图是一棵树 求它拓扑序的个数 \(\bmod (10^9+7)\) \(n \in [1,1000]\) 如果你按照拓扑排序的方法来做,那恐怕你已经想 ...

  4. 3167&colon; &lbrack;Heoi2013&rsqb;Sao &lbrack;树形DP&rsqb;

    3167: [Heoi2013]Sao 题意: n个点的"有向"树,求拓扑排序方案数 Welcome to Sword Art Online!!! 一开始想错了...没有考虑一个点 ...

  5. &lbrack;BZOJ3167&rsqb;&lbrack;P4099&rsqb;&lbrack;HEOI2013&rsqb;SAO&lpar;树形DP&rpar;

    题目描述 Welcome to SAO ( Strange and Abnormal Online).这是一个 VR MMORPG, 含有 n 个关卡.但是,挑战不同关卡的顺序是一个很大的问题. 有 ...

  6. HDU-4661 Message Passing 树形DP&comma;排列组合

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4661 题意:有n个人呈树状结构,每个人知道一个独特的消息.每次可以让一个人将他所知的所有消息告诉和他相 ...

  7. &lbrack;HEOI2013&rsqb;SAO&lpar;树上dp&comma;计数&rpar;

    [HEOI2013]SAO (这写了一个晚上QAQ,可能是我太蠢了吧.) 题目说只有\(n-1\)条边,然而每个点又相互联系.说明它的结构是一个类似树的结构,但是是有向边连接的,题目问的是方案个数,那 ...

  8. 洛谷&dollar;P4099&bsol; &lbrack;HEOI2013&rsqb;&bsol; SAO&bsol; dp&dollar;

    正解:树形$dp$ 解题报告: 传送门$QwQ$. 考虑设$f_i$表示点$i$的子树内的拓扑序排列方案数有多少个. 发现这样不好合并儿子节点和父亲节点.于是加一维,设$f_{i,j}$表示点$i$的 ...

  9. P4099 &lbrack;HEOI2013&rsqb;SAO

    P4099 [HEOI2013]SAO 贼板子有意思的一个题---我()竟然没看题解 有一张连成树的有向图,球拓扑序数量. 树形dp,设\(f[i][j]\)表示\(i\)在子树中\(i\)拓扑序上排 ...

随机推荐

  1. Git怎样撤销一次分支的合并Merge

    出现的问题: 某一天,所有的开发文件已经在development分支上,但是这时候,线上出了一个问题,需要及时修复,于是从master分支上,拉了一个bug分支,进行处理,master分支本应合并bu ...

  2. poj 3335(半平面交)

    链接:http://poj.org/problem?id=3335     //大牛们常说的测模板题 ------------------------------------------------- ...

  3. Prim算法POJ1258

    http://poj.org/problem?id=1258 这道题是最简单的一个啦,,,, #include<stdio.h> #include<iostream> #inc ...

  4. PAT 1075&period; PAT Judge &lpar;25&rpar;

    题目地址:http://pat.zju.edu.cn/contests/pat-a-practise/1075 此题主要考察细节的处理,和对于题目要求的正确理解,另外就是相同的总分相同的排名的处理一定 ...

  5. GitHub使用(一) - 新建个人网站

    1.首先进入“仓库Repositories”,点击“新建New”.

  6. jQuery使用简单示例 validate 插件

    摘录自:http://blog.csdn.net/u010320371/article/details/51104783用户登录 用户名 密码 确认密码 <!DOCTYPE html> & ...

  7. Linux清空文件内容

    日志文件太多,需要清空: echo "" > mylog.log

  8. &period;net 调用java service 代理类方法

        通过Svcutil.exe 工具生成代理类调用 1.找到如下地址“C:\Windows\System32\cmd.exe”  命令行工具,右键以管理员身份运行(视系统是否为win7 而定) 2 ...

  9. js分析 汽&lowbar;车&lowbar;之&lowbar;家 js生成css伪元素 hs&lowbar;kw44&lowbar;configUS&colon;&colon;before

    0.参考 https://developer.mozilla.org/zh-CN/docs/Web/CSS/Pseudo-elements https://developer.mozilla.org/ ...

  10. 一脸懵逼学习Zookeeper(动物园管理员)---》高度可靠的分布式协调服务

    1:Zookeeper是一个分布式协调服务:就是为用户的分布式应用程序提供协调服务 A.zookeeper是为别的分布式程序服务的 B.Zookeeper本身就是一个分布式程序(只要有半数以上节点存活 ...