【ZJOI2017】仙人掌





参考博客:https://www.cnblogs.com/wfj2048/p/6636028.html

我们先求出\(dfs\)树(就是\(dfs\)一遍),然后问题就变成了树形\(DP\)。

我们先判断无解:就用定义来判断,如果一条边出现在多个环里面就无解。

然后我们将所有在环上的边拆了,因为这些边不可能再出现在一个新的环中。于是我们得到了一个森林。

我们设\(f_v\)表示以\(v\)为根的子树得到仙人掌的方案数。\(ans=\prod_{v\ is\ root} f_v\)。

首先我们考虑子树之间不连边,容易得到:\(f_v=\prod f_{son_v}\),也就是所有子树\(f\)之积。

然后我们观察子树之间的连边:一个子树最多只会连出去一条边,也就是只会两两配对连边。我们设\(g_v\)表示有\(v\)个子树两两之间连边的方案数,则:

\[g_v=g_{v-1}+(v-1)*g_{v-2}
\]

这个转移的意义是考虑第\(v\)个点连不连边,如果不连,方案数就是其他\(v-1\)个点连边的方案;如果连,就从之前的\(v-1\)个点中任选一个相连,剩下的再连边。

设\(|sn_v|\)表示\(v\)的儿子个数。则\(f_v=g_{|sn_v|}*\prod f_{son_v}\)。

我的理解是:假设\(v\)的两个儿子\(a,b\)的子树之间要连边,设\(e_a,e_b\)分别表示\(a\)到\(v\)和\(b\)到\(v\)的边,那么我们一定是将之前覆盖了\(e_a\)和\(e_b\)的两个环的起点(深度最深的那个点,如果没有,就是\(a,b\))连接在一起。所以答案是\(g_{|sn_v|}\)乘上所有子树中连边的方案数。

但是\(v\)的子树中还可以向\(v\)的父亲连边。我们就把\(v\)的父亲也看做一个子节点。所以:

\[\begin{cases}
f_v=g_{|sn_v|}*\prod f_{sn_v}\ (v\ is\ not\ root)\\
f_v=g_{|sn_v|+1}*\prod f_{sn_v}\ (v\ is\ root)
\end{cases}
\]

代码:

#include<bits/stdc++.h>
#define ll long long
#define N 1000005 using namespace std;
inline int Get() {int x=0,f=1;char ch=getchar();while(ch<'0'||ch>'9') {if(ch=='-') f=-1;ch=getchar();}while('0'<=ch&&ch<='9') {x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}return x*f;} const ll mod=998244353;
int n,m;
int x[N],y[N];
struct graph {
int cnt,to[N<<2],nxt[N<<2];
int h[N<<1];
void add(int i,int j) {
to[++cnt]=j;
nxt[cnt]=h[i];
h[i]=cnt;
}
void Init() {
cnt=0;
for(int i=1;i<=n;i++) h[i]=0;
}
}s; ll f[N],g[N];
int dfn[N],id;
int dep[N],fa[N];
int tim[N];
void dfs(int v) {
dfn[v]=++id;
for(int i=s.h[v];i;i=s.nxt[i]) {
int to=s.to[i];
if(dfn[to]) continue ;
fa[to]=v;
dep[to]=dep[v]+1;
dfs(to);
}
} bool cmp(int a,int b) {return dep[a]<dep[b];} bool vis[N];
void DP(int v,int flag) {
vis[v]=1;
f[v]=1;
int sn=0;
for(int i=s.h[v];i;i=s.nxt[i]) {
int to=s.to[i];
if(tim[to]>1) continue ;
if(to==fa[v]||fa[to]!=v) continue ;
sn++;
DP(to,0);
f[v]=f[v]*f[to]%mod;
}
if(flag) f[v]=f[v]*g[sn]%mod;
else f[v]=f[v]*g[sn+1]%mod;
} int st[N];
void work() {
for(int i=1;i<=m;i++) {
int a=x[i],b=y[i];
if(dfn[a]<dfn[b]) swap(a,b);
while(a!=b) {
tim[a]++;
if(tim[a]>2) {
cout<<0<<"\n";
return ;
}
a=fa[a];
}
}
for(int i=1;i<=n;i++) st[i]=i;
sort(st+1,st+1+n,cmp);
ll ans=1;
for(int i=1;i<=n;i++) {
int now=st[i];
if(vis[now]) continue ;
DP(now,1);
ans=ans*f[now]%mod;
}
cout<<ans<<"\n";
} void Init() {
s.Init();
for(int i=1;i<=n;i++) vis[i]=tim[i]=fa[i]=dfn[i]=dep[i]=0;
id=0;
} int main() {
g[0]=g[1]=1;
for(int i=2;i<=500005;i++) g[i]=(g[i-1]+g[i-2]*(i-1))%mod;
int T=Get();
while(T--) {
n=Get(),m=Get();
Init();
for(int i=1;i<=m;i++) {
x[i]=Get(),y[i]=Get();
s.add(x[i],y[i]),s.add(y[i],x[i]);
}
dep[1]=1;
dfs(1);
work();
}
return 0;
}

【ZJOI2017】仙人掌的更多相关文章

  1. 【BZOJ4784】[ZJOI2017]仙人掌(Tarjan,动态规划)

    [BZOJ4784][ZJOI2017]仙人掌(Tarjan,动态规划) 题面 BZOJ 洛谷 题解 显然如果原图不是仙人掌就无解. 如果原图是仙人掌,显然就是把环上的边给去掉,变成若干森林连边成为仙 ...

  2. [BZOJ4784][ZJOI2017]仙人掌(树形DP)

    4784: [Zjoi2017]仙人掌 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 312  Solved: 181[Submit][Status] ...

  3. bzoj4784 [Zjoi2017]仙人掌

    Description 如果一个无自环无重边无向连通图的任意一条边最多属于一个简单环,我们就称之为仙人掌.所谓简单环即不经过重复的结点的环. 现在九条可怜手上有一张无自环无重边的无向连通图,但是她觉得 ...

  4. ●洛谷P3687 [ZJOI2017]仙人掌

    题链: https://www.luogu.org/problemnew/show/P3687题解: 计数DP,树形DP. (首先对于这个图来说,如果初始就不是仙人掌,那么就直接输出0) 然后由于本来 ...

  5. 【做题】ZJOI2017仙人掌——组合计数

    原文链接 https://www.cnblogs.com/cly-none/p/ZJOI2017cactus.html 给出一个\(n\)个点\(m\)条边的无向连通图,求有多少种加边方案,使得加完后 ...

  6. LOJ2250 [ZJOI2017] 仙人掌【树形DP】【DFS树】

    题目分析: 不难注意到仙人掌边可以删掉.在森林中考虑树形DP. 题目中说边不能重复,但我们可以在结束后没覆盖的边覆盖一个重复边,不改变方案数. 接着将所有的边接到当前点,然后每两个方案可以任意拼接.然 ...

  7. zjoi2017 仙人掌

    题解: 好难的dp啊...看题解看了好久才看懂 http://blog.csdn.net/akak__ii/article/details/65935711 如果一开始的图就不是仙人掌,答案显然为0, ...

  8. 2019.02.07 bzoj4784: [Zjoi2017]仙人掌(仙人掌+树形dp)

    传送门 题意:给一个无向连通图,问给它加边形成仙人掌的方案数. 思路: 先考虑给一棵树加边形成仙人掌的方案数. 这个显然可以做树形dp. fif_ifi​表示把iii为根的子树加边形成仙人掌的方案数. ...

  9. BZOJ4784 ZJOI2017仙人掌(树形dp+dfs树)

    首先考虑是棵树的话怎么做.可以发现相当于在树上选择一些长度>=2的路径使其没有交,同时也就相当于用一些没有交的路径覆盖整棵树. 那么设f[i]为覆盖i子树的方案数.转移时考虑包含根的路径.注意到 ...

  10. 【题解】ZJOI2017仙人掌

    感觉这题很厉害啊,虽然想了一天多但还是失败了……(:д:) 这题首先注意到给定图中如果存在环其实对于答案是没有影响的.然后关键之处就在于两个 \(dp\) 数组,其中 \(f[u]\) 表示以 \(u ...

随机推荐

  1. nginx配置虚拟机

    在/usr/local/nginx/conf目录下nginx.conf文件是nginx的配置文件. 一.通过端口号区分虚拟机 在nginx.conf文件中添加一个Service节点,修改端口号: se ...

  2. 【Java每日一题】20170221

    20170220问题解析请点击今日问题下方的“[Java每日一题]20170221”查看(问题解析在公众号首发,公众号ID:weknow619) package Feb2017; public cla ...

  3. 【Spring】29、SpringBoot中@SpringBootApplication的使用

    之前用户使用的是3个注解注解他们的main类.分别是@Configuration,@EnableAutoConfiguration,@ComponentScan.由于这些注解一般都是一起使用,spri ...

  4. JavaWeb学习日记----XML的解析

    XML的解析简介: 在学习JavaScript时,我们用的DOM来解析HEML文档,根据HTML的层级结构在内存中分配一个树形结构,把HTML的标签啊,属性啊和文本之类的都封装成对象. 比如:docu ...

  5. 三角形(hdu1249)递推

    三角形 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submiss ...

  6. 使用laravel框架开发接口时ajax post请求报错419

    nginx服务器,使用laravel框架开发后台接口.get请求正常,但是post请求一直报错.H5和APP都不成功,code=419. 解决办法: 找到 VerifyCsrfToken.php文件( ...

  7. SQL 读取XML到Datatable

    DECLARE @hdoc INT --XML 数据格式 --------------------------------------------------------- ) SET @doc = ...

  8. spring一些简单小注意知识点

    Spring 配置详解 <!-- Bean元素:使用该元素描述需要spring容器管理的对象            class属性:被管理对象的完整类名.            name属性:给 ...

  9. GDPR 和个人信息保护的小知识

    从2018年5月25日起,欧盟的<通用数据保护条例>(简称 GDPR,General Data Protection Regulation)开始强制施行.这个规范加强了对个人信息的保护,并 ...

  10. 自定义View类

    一.如何创建自定义的View类 ①.创建一个继承android.view.View类的Java类,并且重写构造方法(至少需要重写一个构造方法) ②.根据需要重写其他方法 ③.在项目的活动中,创建并实例 ...