P4099 [HEOI2013]SAO


贼板子有意思的一个题~~~我()竟然没看题解

有一张连成树的有向图,球拓扑序数量。

树形dp,设\(f[i][j]\)表示\(i\)在子树中\(i\)拓扑序上排名为\(j\)的方案数。

难就难在转移,现在有两个树\(x\)和\(y\),其中\(x\)是父亲,\(x\)的拓扑序小于\(y\)的,从\(f[x][p1],f[y][p2]\)转移到\(newf[x][p3]\):\(x\)在原序列中排名\(p1\),新序列中\(p3\);\(y\)在原序列中排名\(p2\),新序列中\(p4\),那么限制是\(p3<p4\)

那么\(x\)子树中,\(p1\)左边的点也一定在\(p3\)左边(因为是和同一个点\(x\)比较)

又有限制\(p3<p4\),所以\(y\)的原序列中排名为\([p2,siz_y]\)都在\(p3\)右边,\([1,p2-1]\)可以有一些在\(p3\)右边,有一些在左边

所以\(p3\)的左边数的数量限制是:\(p1-1\leq p3-1\leq p1-1+p2-1\),也就是\(p1\leq p3\leq p1+p2-1\)

\(p3\)的范围就确定了,转移当然还要乘组合数,先考虑左边的情况,左边有\(p3-1\)个点,一定有\(p1-1\)个来自\(x\)的原序列,所以左边的方案数为\(C_{p3-1}^{p1-1}\);右边同理,有\(siz_x+siz_y-p3\)个点,一定有\(siz_x-p1\)个点来自\(x\)的原序列,所以右边方案数是\(C_{siz_x+siz_y-p3}^{siz_x-p1}\)。

综上,从\(f[x][p1],f[y][p2]\)转移到\(newf[x][p3]\),而且新序列中\(x\)在\(y\)左边,要满足\(p1\leq p3\leq p1+p2-1\),转移方程是

\(newf[x][p3]+=C_{p3-1}^{p1-1}C_{siz_x+siz_y-p3}^{siz_x-p1}f[x][p1]f[y][p2]\)

还有一种情况就是新序列中\(x\)在\(y\)右边的,也是同理推,转移方程一样,唯一的区别是\(p3\)的取值范围是\(p1+p2\leq p3\leq p1+siz_x\)。

然后就解决了,然而是\(O(n^3)\)的,考虑优化

当然把式子写出来啊(还是以新序列中\(x\)在\(y\)左边为例)

for p1 in [1,siz_x]
for p2 in [1,siz_y]
for p3 in [p1,p1+p2-1]
转移

观察一下转移方程,好像\(p2\)只出现了一次,还是连续的,于是调换循环顺序(省去对循环范围的推倒):

for p1 in [1,siz_x]
for p3 in [p1,p1+siz_y-1]
for p2 in [p3-p1+1,siz_y]
转移

那么把循环转移成了p2最后一个循环,就可以用前缀和优化转移了,然后这题切了。。。

就是新序列中\(x\)在\(y\)右边的情况直接看代码。

#include<bits/stdc++.h>
#define il inline
#define vd void
#define mod 1000000007
typedef long long ll;
il ll gi(){
ll x=0,f=1;
char ch=getchar();
while(!isdigit(ch)){
if(ch=='-')f=-1;
ch=getchar();
}
while(isdigit(ch))x=x*10+ch-'0',ch=getchar();
return x*f;
}
int fir[1010],dis[2010],nxt[2010],w[2010],id;
il vd link(int a,int b,int c){nxt[++id]=fir[a],fir[a]=id,dis[id]=b,w[id]=c;}
int f[1010][1010],g[1010],siz[1010];
int C[1010][1010];
il vd dfs(int x){
siz[x]=1;f[x][1]=1;
for(int i=fir[x];i;i=nxt[i]){
if(siz[dis[i]])continue;
dfs(dis[i]);
memcpy(g,f[x],sizeof g);
memset(f[x],0,sizeof f[x]);
if(w[i]==1){
for(int p1=1;p1<=siz[x];++p1)
for(int p3=p1;p3<p1+siz[dis[i]];++p3)
f[x][p3]=(f[x][p3]+1ll*C[siz[x]+siz[dis[i]]-p3][siz[x]-p1]*C[p3-1][p1-1]%mod*g[p1]%mod*(f[dis[i]][siz[dis[i]]]-f[dis[i]][p3-p1]+mod))%mod;
}else{
for(int p1=1;p1<=siz[x];++p1)
for(int p3=p1+1;p3<=p1+siz[dis[i]];++p3)
f[x][p3]=(f[x][p3]+1ll*C[siz[x]+siz[dis[i]]-p3][siz[x]-p1]*C[p3-1][p1-1]%mod*g[p1]%mod*f[dis[i]][p3-p1])%mod;
}
siz[x]+=siz[dis[i]];
}
for(int i=1;i<=siz[x];++i)f[x][i]=(f[x][i]+f[x][i-1])%mod;
}
int main(){
#ifdef XZZSB
freopen("in.in","r",stdin);
freopen("out.out","w",stdout);
#endif
C[0][0]=1;
for(int i=1;i<=1000;++i){
C[i][0]=1;
for(int j=1;j<=i;++j)C[i][j]=(C[i-1][j-1]+C[i-1][j])%mod;
}
int T=gi();
while(T--){
id=0;memset(fir,0,sizeof fir);memset(siz,0,sizeof siz);
int n=gi(),a,b;char ch;
for(int i=1;i<n;++i){
scanf("%d %c %d",&a,&ch,&b);++a,++b;
link(a,b,ch=='<');link(b,a,ch=='>');
}
dfs(1);
printf("%d\n",f[1][n]);
}
return 0;
}

P4099 [HEOI2013]SAO的更多相关文章

  1. P4099 [HEOI2013]SAO(树形dp)

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

  2. [BZOJ3167][P4099][HEOI2013]SAO(树形DP)

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

  3. 洛谷 P4099 - [HEOI2013]SAO(树形 dp)

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

  4. luogu P4099 [HEOI2013]SAO

    传送门 吐槽题目标题 这个依赖关系是个树,可以考虑树型dp,设f_i表示子树i的答案 因为这是个序列问题,是要考虑某个数的位置的,所以设\(f_{i,j}\)表示子树i构成的序列,i在第j个位置的方案 ...

  5. 洛谷P4099 [HEOI2013]SAO(树形dp)

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

  6. 洛谷$P4099\ [HEOI2013]\ SAO\ dp$

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

  7. 【做题记录】 [HEOI2013]SAO

    P4099 [HEOI2013]SAO 类型:树形 \(\text{DP}\) 这里主要补充一下 \(O(n^3)\) 的 \(\text{DP}\) 优化的过程,基础转移方程推导可以参考其他巨佬的博 ...

  8. 3167: [Heoi2013]Sao [树形DP]

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

  9. 【BZOJ3167】[HEOI2013]SAO(动态规划)

    [BZOJ3167][HEOI2013]SAO(动态规划) 题面 BZOJ 洛谷 题解 显然限制条件是一个\(DAG\)(不考虑边的方向的话就是一棵树了). 那么考虑树型\(dp\),设\(f[i][ ...

随机推荐

  1. EF Core扩展工具记录

    Microsoft.EntityFrameworkCore.AutoHistory Microsoft.EntityFrameworkCore 的一个插件,支持自动记录数据更改历史记录. GitHub ...

  2. javascript获取DOM对象三种方法

    1. getElementByID() getElementByID()方法可返回对拥有指定ID的第一个对象的引用 2. getElementByTagName() getElementByTagNa ...

  3. 4.93Python数据类型之(8)集合

    目录 目录 前言 (一)基本概念 ==1.1有序于无序== ==1.2是否随机访问== ==1.3重复性== ==1.4可变与不可变的集合== (二)集合的增删改查 ==2.1集合的增加== ==2. ...

  4. rls与rlsd

    服务器端的程序一般有如下几个过程,首先是bind,然后再是listen,最后是accept.再往后就是客户端与服务器连接后的各种操作了. 相比之下,客户端的程序就比较简单了,只需先获得sock_id, ...

  5. Axure RP Pro7.0的key注册码加汉化非破解

    上次我们刚分享过Axure RP Pro6.5 key注册码加汉化非破解,我还要分享一个Axure RP Pro7.0的key注册码加汉化,非破解哦. 当然方法还是不变,先用下面的密钥激活.用户名就是 ...

  6. rowid快速分页解析

    版权声明:个人随笔,实用你就COPY,看不懂不解释 https://blog.csdn.net/HelloCqk1/article/details/36628787 --分页第一步 获取数据物理地址 ...

  7. linux命令之 df file fsck fuser

    有非常多人说,网上非常多知识点都有了.为什么你还要在自己的博客中反复这些东西呢? 我想说的是.别人写的东西是别人理解的东西,同一时候也是别人学习过程的总结,对于自己来说.自己写自己的博客最基本的目的就 ...

  8. Hunter -- 批量文件管理工具

    一个简单工具, 用来进行批量文件的重命名, 更改后缀, 搜索特定后缀, 等 下载地址:http://download.csdn.net/download/kuangsun/7545179

  9. Spring 注解大全

    @Autowired 自动注入 (存在多个可注入Bean时,通过 @Qualifier 指定)@Resource 与@Autowired作用相同@Repository 只能标注在 DAO 类上.该注解 ...

  10. 【转】这五种方法前四种方法只支持IE浏览器,最后一个方法支持当前主流的浏览器(火狐,IE,Chrome,Opera,Safari)

    <!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF-8&quo ...