HEOI2013SAO
题目描述
给定一个\(DAG\),问这个\(DAG\)有多少种拓扑序。
题解
我们首先需要设计一个能够比较好的转移的状态。
我们可以设\(dp[i][j]\)表示第i个点在当前\(dp\)的子图中拓扑排名为\(j\)的方案数。
至于\(dp\)的方式,我们发现只有\(n-1\)条边,所以我们并不用在\(DAG\)上\(dp\),直接在建出来的树上\(dp\)就好了。
转移的话,对于一条树边\(u->v\)我们先枚举转移之后的\(u\)的排名,再枚举当前\(v\)的排名,我们发现还需要枚举\(v\)的子树在\(u\)排名之前的点的个数\(x\)。
当\(u\)需要在\(v\)之后时:
\]
发现转移是一个后缀和的形式,可以用后缀和优化)。
当\(u\)需要在\(v\)之前时。
\]
转移是一个前缀和的形式,也可以用前缀和优化。
经过优化后,这个\(dp\)的复杂度可以优化到树形背包的复杂度,也就是\(O(n^2)\)。
代码
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#define N 1009
using namespace std;
typedef long long ll;
const int maxn=1000;
const int mod=1000000007;
char s[1];
ll ans,dp[N][N],now[N],c[N][N],g[N];
int tot,head[N],size[N],n;
inline ll rd(){
ll x=0;char c=getchar();bool f=0;
while(!isdigit(c)){if(c=='-')f=1;c=getchar();}
while(isdigit(c)){x=(x<<1)+(x<<3)+(c^48);c=getchar();}
return f?-x:x;
}
inline void MOD(ll &x){x=x>=mod?x-mod:x;}
struct edge{int n,to,l;}e[N<<1];
inline void add(int u,int v,int l){e[++tot].n=head[u];e[tot].to=v;head[u]=tot;e[tot].l=l;}
void dfs(int u,int fa){
size[u]=1;
dp[u][1]=1;
for(int i=head[u];i;i=e[i].n){
int v=e[i].to;if(v==fa)continue;
dfs(v,u);
if(e[i].l){
for(int j=1;j<=size[u]+size[v];++j){
int ma=min(size[v],j-1);
now[ma+1]=0;
for(int x=ma;x>=1;--x)
MOD(now[x]=now[x+1]+dp[u][j-x]*c[j-1][x]%mod*c[size[u]-(j-x)+size[v]-x][size[v]-x]%mod);
for(int k=1;k<=size[v];++k)MOD(g[j]+=dp[v][k]*now[k]%mod),now[k]=0;
}
}
else{
for(int j=1;j<=size[u]+size[v];++j){
int ma=min(size[v],j-1);
for(int x=0;x<=ma;++x)
MOD(now[x]=now[x-1]+dp[u][j-x]*c[j-1][x]%mod*c[size[u]-(j-x)+size[v]-x][size[v]-x]%mod);
for(int k=1;k<=size[v];++k)MOD(g[j]+=now[min(k-1,ma)]*dp[v][k]%mod);
for(int x=0;x<=ma;++x)now[x]=0;
}
}
size[u]+=size[v];
for(int j=1;j<=size[u];++j)dp[u][j]=g[j],g[j]=0;
}
}
int main(){ // 1 -> dayu 0 -> xiaoyu
int T=rd();
c[0][0]=1;
for(int i=1;i<=maxn;++i){
c[i][0]=1;
for(int j=1;j<=i;++j)MOD(c[i][j]=c[i-1][j]+c[i-1][j-1]);
}
while(T--){
n=rd();int u,v;
memset(dp,0,sizeof(dp));
memset(head,0,sizeof(head));tot=0;
for(int i=1;i<n;++i){
u=rd()+1;scanf("%s",s);v=rd()+1;
add(u,v,s[0]=='>');add(v,u,s[0]=='<');
}
ans=0;dfs(1,0);
for(int i=1;i<=n;++i)MOD(ans+=dp[1][i]);
printf("%lld\n",ans);
}
return 0;
}
HEOI2013SAO的更多相关文章
- BZOJ3167/BZOJ4824 HEOI2013SAO/CQOI2017老C的键盘(树形dp)
前者是后者各方面的强化版. 容易想到设f[i][j]表示i子树中第j小的是i的方案数(即只考虑相对关系).比较麻烦的在于转移.考虑逐个合并子树.容易想到枚举根原来的排名和子树根原来的排名,算一发组合数 ...
- CodeChef LEMOVIE
题意:给你n个数字(下标不同数值相同的数字应当被认为是不同的数字),有n!种排列方式.每种排列方式的价值定义为:第一次出现时比前面的所有数字都大的数值个数. 比如1,2,2,3这个排列中,1,2,3这 ...
随机推荐
- WPF ResourceDictionary 主题资源替换(一)
当我们需要在程序中替换主题,更换另一套背景.颜色.样式时,如何在不修改资源Key值,直接替换呢? 问题&疑问 1. Key值冲突 同一ResourceDictionary中,不可以使用相同Ke ...
- c# Base64解密加密
private static string base64EncodeChars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz ...
- 最新阿里Java技术面试题,看这一文就够了!
金三银四跳槽季即将到来,作为 Java 开发者你开始刷面试题了吗?别急,小编整理了阿里技术面试题,看这一文就够了! 阿里面试题目目录 技术一面(基础面试题目) 技术二面(技术深度.技术原理) 项目实战 ...
- jenkins实现以gitlab为代码仓库的构建
简介 前一篇随笔是安装jenkins的过程,比较简单,这一次说一下用jenkins配置以gitlab为代码管理仓库的maven项目的完整个构建过程,以及我碰到的一些问题.由于是maven项目,所以我们 ...
- php禁用函数设置及查看方法详解
这篇文章主要介绍了php禁用函数设置及查看方法,结合实例形式分析了php禁用函数的方法及使用php探针查看禁用函数信息的相关实现技巧,需要的朋友可以参考下 本文实例讲述了php禁用函数设置及查看方法. ...
- off-canvas:抽屉式页面布局的纯css实现
Off-canvas即抽屉式的侧边导航栏布局,导航栏在大尺寸屏幕的时候可以设置无需隐藏,小尺寸屏幕的时候自动隐藏,并出现.off-canvas-toggle用以打开导航栏,打开导航栏的状态下可以点击非 ...
- Android远程桌面助手(B1371)
Android远程桌面助手(B1371),下载:https://files.cnblogs.com/files/we-hjb/ARDC%28B1371%29.7z 1.增加了对超大分辨率4320*21 ...
- 什么是Docker,它可干什么?
定义我们知道,软件依赖的环境大致包括: 1• 配置文件2• 代码3• tomcat4• JDK5• 操作系统 Docker作为一个软件集装箱化平台,可以让开发者构建应用程序时,将它与其依赖环境一起打包 ...
- 通过 Sqoop1.4.7 将 Mysql5.7、Hive2.3.4、Hbase1.4.9 之间的数据导入导出
目录 目录 1.什么是 Sqoop? 2.下载应用程序及配置环境变量 2.1.下载 Sqoop 1.4.7 2.2.设置环境变量 2.3.设置安装所需环境 3.安装 Sqoop 1.4.7 3.1.修 ...
- d3.csv()后获取的数据不是数组,而是对象
我的csv文件: year,population 1953,5.94 1964,6.95 1982,10.08 1990,11.34 2000,12.66 2010,13.40 使用d3.csv()输 ...