[洛谷P8867] [NOIP2022] 建造军营

[NOIP2022] 建造军营

题目描述

A 国与 B 国正在激烈交战中,A 国打算在自己的国土上建造一些军营。

A 国的国土由 n 座城市组成,m 条双向道路连接这些城市,使得任意两座城市均可通过道路直接或间接到达。A 国打算选择一座或多座城市(至少一座),并在这些城市上各建造一座军营。

众所周知,军营之间的联络是十分重要的。然而此时 A 国接到情报,B 国将会于不久后袭击 A 国的一条道路,但具体的袭击目标却无从得知。如果 B 国袭击成功,这条道路将被切断,可能会造成 A 国某两个军营无法互相到达,这是 A 国极力避免的。因此 A 国决定派兵看守若干条道路(可以是一条或多条,也可以一条也不看守),A 国有信心保证被派兵看守的道路能够抵御 B 国的袭击而不被切断。

A 国希望制定一个建造军营和看守道路的方案,使得 B 国袭击的无论是 A 国的哪条道路,都不会造成某两座军营无法互相到达。现在,请你帮 A 国计算一下可能的建造军营和看守道路的方案数共有多少。由于方案数可能会很多,你只需要输出其对 1,000,000,007(109+7) 取模的值即可。两个方案被认为是不同的,当且仅当存在至少一 座城市在一个方案中建造了军营而在另一个方案中没有,或者存在至少一条道路在一个 方案中被派兵看守而在另一个方案中没有。

输入格式

第一行包含两个正整数 n,m,分别表示城市的个数和双向道路的数量。

接下来 m 行,每行包含两个正整数 ui,vi,描述一条连接 uivi 的双向道路。保证没有重边和自环。

输出格式

输出一行包含一个整数,表示建造军营和看守道路的方案数对 1,000,000,007(109+7) 取模的结果。

样例 #1

样例输入 #1

2 1
1 2

样例输出 #1

5

样例 #2

样例输入 #2

4 4
1 2
2 3
3 1
1 4

样例输出 #2

184

样例 #3

样例输入 #3

见附加文件里的 barrack/barrack3.in

样例输出 #3

见附加文件里的 barrack/barrack3.ans

样例 #4

样例输入 #4

见附加文件里的 barrack/barrack4.in

样例输出 #4

见附加文件里的 barrack/barrack4.ans

提示

样例 1 解释

A 国有两座城市,一条道路连接他们。所有可能的方案如下:

  • 在城市 1 建军营, 不看守这条道路;
  • 在城市 1 建军营, 看守这条道路;
  • 在城市 2 建军营, 不看守这条道路;
  • 在城市 2 建军营, 看守这条道路;
  • 在城市 1,2 建军营, 看守这条道路。

数据规模与约定

对所有数据,保证 1n5×105n1m1061ui,vinuivi

各测试点的信息如下

测试点编号 n m 特殊条件
13 8 10
47 16 25
89 3000 5000
1011 5×105 106 特殊性质 A
1214 5×105 106 m=n1
1516 5×105 106 m=n
1720 5×105 106

特殊性质 A:保证 m=n1 且第 i 条道路连接城市 ii+1

试想,如果两个城市在一个边双连通分块里面,那么肯定没有办法通过删除一条边来使这两个城市分隔两地。所以可以考虑边双连通分块。

先把双连通分块缩成一个点,缩完后就是一棵树。这棵树中我们可以选任意个点,任意个边,要求所有点必须被联通的方案数。一看就是树形dp.

定义 dpi,0i 的子树中一定有选点,子树外没有选点时的方案数,dpi,1i 的子树中一定有选点,子树外有选点的方案数。为什么要控制子树外和子树外呢?因为这会影响到边是否必选的情况。钦定根为1,那么最终答案为 dp1,0。同时可以发现,如果一棵子树中没点,那么方案数就是当中的边选不选了。

dpx,1 的转移简单,那就先看他吧。设 (x,y) 有边子树外存在点,那么如果 y 中有选择点,那么 (x,y) 这条边必选,然后递归到 dpy 计算。如果 a 是所有的子树的方案数,b为所有子树都不选点的方案数。如果 x 为关键点,方案数为 a。否则就要在子树中至少选一个点,方案数 ab(要减去一个点都不选的)。注意我们上面进行了一个缩点,所以如果某一个点如果要将他选为关键点,那么方案也有 2cc 为这个边双的点的数量。

来看 dpx,0 的转移,那么此时子 x 树内必须选点,那要分成子树内只有某一棵子树有点的情况和子树内有多棵子树有点的情况。如果只有一棵子树有点,那么递归到 dpy,0 计算,否则递归到 dpy,1 计算。最后注意要 dpy,1 要减去只选某一棵子树或者没选点的情况。

#include<bits/stdc++.h>
using namespace std;
typedef unsigned long long ULL;
const int N=5e5+5,M=2e6+5,P=1e9+7;
int n,m,u,v,e_num=1,hd[N],dfn[N],low[N],id[N],h[N],g_num,idx,tme,cnt,sz[N],c[N],pw[M],dp[N][2],ans;
struct edge{
	int u,v,nxt,f;
}e[M],g[M];
void add_edge(int u,int v)
{
	e[++e_num]=(edge){u,v,hd[u],0};
	hd[u]=e_num;
}
void addedge(int u,int v)
{
//	printf("%d %d\n",u,v);
	g[++g_num]=(edge){u,v,h[u],0};
	h[u]=g_num;
}
void tarjan(int x,int y)
{
	dfn[x]=low[x]=++idx;
	for(int i=hd[x];i;i=e[i].nxt)
	{
		if(e[i].v!=y)
		{
			if(!dfn[e[i].v])
			{
				tarjan(e[i].v,x);
				low[x]=min(low[x],low[e[i].v]);
			}
			else 
				low[x]=min(low[x],dfn[e[i].v]);
			if(low[e[i].v]>dfn[x])
				e[i].f=e[i^1].f=1;
		}
	}
}
void sou(int x,int tme)
{
	id[x]=tme,c[tme]++;
	for(int i=hd[x];i;i=e[i].nxt)
		if(!e[i].f&&!id[e[i].v])
			sou(e[i].v,tme);
}
void dfs(int x,int y)
{
	sz[x]=1;
	int a=1,b=1,d=0,e=0,f=1;
	for(int i=h[x];i;i=g[i].nxt)
	{
		if(g[i].v!=y)
		{
			dfs(g[i].v,x);
			sz[x]+=sz[g[i].v];
		}
	}
//	printf("%d\n",d);
	for(int i=h[x];i;i=g[i].nxt)
	{
		if(g[i].v!=y)
		{
			a=1LL*a*(pw[sz[g[i].v]]+dp[g[i].v][1])%P;
			b=1LL*b*pw[sz[g[i].v]]%P;
			(d+=1LL*pw[sz[x]-sz[g[i].v]-1]*dp[g[i].v][1]%P)%=P;
			(e+=1LL*pw[sz[x]-sz[g[i].v]]%P*dp[g[i].v][0]%P)%=P;
//			f=1LL*f*(pw[sz[g[i].v]]+dp[g[i].v][1]);
		}
	}
//	printf("%d\n",d);
	dp[x][1]=(((a-b)+1LL*(pw[c[x]]-1)*a%P)%P+P)%P;
	dp[x][0]=(1LL*(pw[c[x]]-1)*a%P+((a-b-d)%P+P)%P+e)%P;
//	printf("%d\n",d);
//	printf("%d %d %d\n",x,dp[x][0],dp[x][1]);
//	printf("%d %d %d %d %d %d %d %d\n",x,dp[x][0],dp[x][1],a,f,b,d,e);
}
int main()
{
//	freopen("barrack.in","r",stdin);
//	freopen("barrack.out","w",stdout);
	scanf("%d%d",&n,&m);
	for(int i=pw[0]=1;i<=m;i++)
		pw[i]=pw[i-1]*2%P;
	for(int i=1;i<=m;i++)
	{
		scanf("%d%d",&u,&v);
		add_edge(u,v);
		add_edge(v,u);
	}
//	printf("%d\n",e_num);
	for(int i=1;i<=n;i++)
		if(!dfn[i])
			tarjan(i,0);
	for(int i=1;i<=n;i++)
	{
		if(!id[i])
		{
			++tme;
			sou(i,tme);
		}
	}
	for(int i=2;i<=e_num;i+=2)
	{
		if(id[e[i].u]==id[e[i].v])
			++cnt;
		else
		{
			addedge(id[e[i].u],id[e[i].v]);
			addedge(id[e[i].v],id[e[i].u]);
		}
	}
//	printf("%d\n",cnt);
	dfs(1,0);
//	printf("%llu %llu\n",pw[cnt],dp[1]+pw[tme-1]-1);
//	printf("%d\n",pw[cnt]);
//	printf("%d\n",cnt);
	printf("%d",(1LL*pw[cnt]*dp[1][0]%P));
	return 0;
}
posted @   灰鲭鲨  阅读(557)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 单线程的Redis速度为什么快?
· SQL Server 2025 AI相关能力初探
· AI编程工具终极对决:字节Trae VS Cursor,谁才是开发者新宠?
· 展开说说关于C#中ORM框架的用法!
点击右上角即可分享
微信分享提示