传送门

题意:给一个无向连通图,问给它加边形成仙人掌的方案数。


思路:

先考虑给一棵树加边形成仙人掌的方案数。

这个显然可以做树形dp。

fif_ifi​表示把iii为根的子树加边形成仙人掌的方案数。

然后有两种情况:

  1. iii点没有父亲
  2. iii点有父亲

对于第一种情况即iii是树根的情况,显然fi=(∏fv)∗g∣sonp∣f_i=(\prod f_v)*g_{|son_p|}fi​=(∏fv​)∗g∣sonp​∣​,其中gig_igi​表示给iii个儿子两两配对(每个儿子可配可不配的方案数)。

对于第二种情况有可能把iii跟父亲连上的那条边拿来放进一个环里,因此把iii也看成一个允许配对的连通块即可,则fi=(∏fv)∗g∣sonp∣+1f_i=(\prod f_v)*g_{|son_p|+1}fi​=(∏fv​)∗g∣sonp​∣+1​

现在只需要预处理出ggg数组即可,我们再次用dpdpdp解决这个问题:

g0=g1=1,gi=gi−1+(i−1)gi−2,(i≥2)g_0=g_1=1,g_i=g_{i-1}+(i-1)g_{i-2},(i\ge2)g0​=g1​=1,gi​=gi−1​+(i−1)gi−2​,(i≥2)

这个递推很简单。

于是我们就成功处理出了树的情况,现在只用考虑原图怎么搞。

  1. 原图不是一个仙人掌,puts(0)puts(0)puts(0)即可
  2. 原图是一个仙人掌,我们把所有在环上面的边都删掉就成了一个森林,把每棵树的方案数统计出来乘起来就是答案。

代码:

#include<bits/stdc++.h>
#define ri register int
using namespace std;
inline int read(){
	int ans=0;
	char ch=getchar();
	while(!isdigit(ch))ch=getchar();
	while(isdigit(ch))ans=(ans<<3)+(ans<<1)+(ch^48),ch=getchar();
	return ans;
}
const int N=5e5+5,mod=998244353;
typedef long long ll;
inline int add(const int&a,const int&b){return a+b>=mod?a+b-mod:a+b;}
inline int dec(const int&a,const int&b){return a>=b?a-b:a-b+mod;}
inline int mul(const int&a,const int&b){return (ll)a*b%mod;}
int n,m,dfn[N],low[N],tot=0,cnt[N],g[N],f[N],fa[N],siz;
bool vis[N],flag;
vector<int>e[N];
map<int,int>ban[N];
inline void solve(int rt,int p){
	ban[rt][p]=ban[p][rt]=1;
	while(p!=rt){
		if(++cnt[p]==2){flag=0;return;}
		ban[p][fa[p]]=ban[fa[p]][p]=1,p=fa[p];
	}
}
void tarjan(int p){
	++siz;
	if(!flag)return;
	dfn[p]=low[p]=++tot;
	for(ri i=0,v;i<e[p].size();++i){
		if((v=e[p][i])==fa[p])continue;
		if(!flag)return;
		if(!dfn[v])fa[v]=p,tarjan(v),low[p]=min(low[p],low[v]);
		else low[p]=min(low[p],low[v]);
	}
	if(!flag)return;
	for(ri i=0,v;i<e[p].size();++i)if(fa[v=e[p][i]]!=p&&dfn[p]<dfn[v])solve(p,v);
}
void dfs(int p,int fat){
	vis[p]=1;
	int du=0,mult=1;
	for(ri i=0,v;i<e[p].size();++i){
		if((v=e[p][i])==fat||ban[p][v])continue;
		dfs(v,p),mult=mul(mult,f[v]),++du;
	}
	if(fat)f[p]=mul(mult,g[du+1]);
	else f[p]=mul(mult,g[du]);
}
int main(){
	g[0]=g[1]=1;
	for(ri i=2;i<=500000;++i)g[i]=add(g[i-1],mul(g[i-2],i-1));
	for(ri tt=read(),ans;tt;--tt){
		n=read(),m=read(),flag=1,siz=0;
		for(ri i=1;i<=n;++i)e[i].clear(),cnt[i]=dfn[i]=low[i]=vis[i]=0,ban[i].clear();
		for(ri i=1,u,v;i<=m;++i)u=read(),v=read(),e[u].push_back(v),e[v].push_back(u);
		tarjan(1),ans=1;
		if(!flag||siz<n){puts("0");continue;}
		for(ri i=1;i<=n;++i)if(!vis[i])dfs(i,0),ans=mul(ans,f[i]);
		cout<<ans<<'\n';
	}
	return 0;
}

最新文章

  1. CanvasWebgl项目介绍
  2. 办公OA的登陆界面..
  3. C语言-纸牌计算24点小游戏
  4. js的一些笔记
  5. PHP中json_encode后中文乱码的解决方案
  6. Asp.Net MVC+BootStrap+EF6.0实现简单的用户角色权限管理4
  7. wpf中手风琴控件Accordion编辑模板后控件不正常。
  8. 关于后台管理linkbutton按钮几个重要属性的理解
  9. 获取表单的初始值,模拟placeholder属性
  10. subeclipse 安装
  11. 在JSP中使用CKEditor网页编辑器
  12. solrj6.2异常--Expected mime type application/octet-stream but got text/html.
  13. VMware安装ubuntu,通过/mnt/hgfs 挂载共享Windows系统文件夹
  14. C#隐藏(new)方法和重写(override)方法
  15. 入侵感知系列之webshell检测思路
  16. Spring IOC注入接口多实现解决
  17. Hadoop源码系列(一)FairScheduler申请和分配container的过程
  18. Typescript学习总结之泛型
  19. 报错:Missing type map configuration or unsupported mapping
  20. 关于eclipse 在创建一个新项目时自动出现的appcompat v7如何解决

热门文章

  1. PAT1026 (大模拟)
  2. JAVA 基本数据结构--数组、链表、ArrayList、Linkedlist、hashmap、hashtab等
  3. 安恒7月赛wp
  4. jquery is()和has()方法
  5. c#: 以模态窗口显示于其它进程窗体之前
  6. C++中的字符数组与字符指针
  7. solrj 测试连接 6.6.5solr集群
  8. gearman的持久化,以mysql的方式
  9. Socket(转自 阿里云)
  10. 比特币测试网络搭建以及RPC服务开启-配置注意事项