【图论】Codeforces 711D Directed Roads

本文介绍了一道 Codeforces 平台上的图论问题 711D 的解题思路与代码实现。该问题要求计算通过逆向若干条边使图无环的方案数。文章详细解析了如何利用 DFS 找到每个环及其贡献,并最终求得总方案数。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

题目链接:

  http://codeforces.com/problemset/problem/711/D

题目大意:

  给一张N个点N条有向边的图,边可以逆向。问任意逆向若干条边使得这张图无环的方案数(mod 1e9+7)。

题目思路:

  【图论】

  因为是N条边所以不会有复杂的环,最多只会有若干组一个环加一条链。

  推算得到,一个大小为k的环对答案的贡献是*(2k-2),而长度为k的链对答案的贡献是2k(链不包括环上的)

  用dfs找出每一组环的大小和链的长度,计算答案即可。




//
//by coolxxx
//#include<bits/stdc++.h>
#include<iostream>
#include<algorithm>
#include<string>
#include<iomanip>
#include<map>
#include<stack>
#include<queue>
#include<set>
#include<bitset>
#include<memory.h>
#include<time.h>
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
//#include<stdbool.h>
#include<math.h>
#define min(a,b) ((a)<(b)?(a):(b))
#define max(a,b) ((a)>(b)?(a):(b))
#define abs(a) ((a)>0?(a):(-(a)))
#define lowbit(a) (a&(-a))
#define sqr(a) ((a)*(a))
#define swap(a,b) ((a)^=(b),(b)^=(a),(a)^=(b))
#define mem(a,b) memset(a,b,sizeof(a))
#define eps (1e-8)
#define J 10
#define mod 1000000007
#define MAX 0x7f7f7f7f
#define PI 3.14159265358979323
#define N 200014
using namespace std;
typedef long long LL;
int cas,cass;
int n,m,lll,ans;
LL aans;
LL e[N];
int to[N];
int t[N];
bool mark[N];
LL mi(int x,int y)
{
	LL sum=1;
	while(y)
	{
		if(y&1)sum=(sum*x)%mod;
		x=(x*x)%mod;y>>=1;
	}
	return sum;
}
void dfs(int u)
{
	while(!mark[u])
	{
		mark[u]=1;t[u]=++cas;
		u=to[u];
	}
	if(t[u]<=cass)
		aans=(aans*e[cas-cass])%mod;
	else
	{
		aans=(aans*((e[cas-t[u]+1]-2+mod)%mod))%mod;
		aans=(aans*e[t[u]-1-cass])%mod;
	}
	return;
}
void init()
{
	int i;
	e[0]=1;
	for(i=1;i<N;i++)e[i]=(e[i-1]*2)%mod;
}
int main()
{
	#ifndef ONLINE_JUDGE
//	freopen("1.txt","r",stdin);
//	freopen("2.txt","w",stdout);
	#endif
	int i,j,k;
	init();
//	for(scanf("%d",&cass);cass;cass--)
//	for(scanf("%d",&cas),cass=1;cass<=cas;cass++)
//	while(~scanf("%s",s+1))
	while(~scanf("%d",&n))
	{
		mem(mark,0);
		for(i=1;i<=n;i++)scanf("%d",&to[i]);
		aans=1;cas=0;
		for(i=1;i<=n;i++)
		{
			if(mark[i])continue;
			cass=cas;
			dfs(i);
		}
		printf("%I64d\n",aans);
	}
	return 0;
}
/*
//

//
*/


### Codeforces 平台上的图论问题及教程 #### 图论问题的特点 Codeforces 的题目难度较高,涵盖从入门到专家级的各类算法问题,适合不同水平的参赛者挑战。平台上的题目涉及多个领域,其中包括图论[^2]。 #### 获取图论问题的方法 为了找到特定类型的图论问题,可以访问 Codeforces 官方网站并利用其内置过滤器功能来筛选标签为 "graphs" 或其他相关子类别的题目。这些标签有助于快速定位感兴趣的图论问题[^1]。 #### 教程与学习资源 对于希望深入理解图论概念的学习者来说,在 Codeforces 博客部分经常会有经验丰富的选手分享关于解决复杂图论难题的文章和技术贴士;此外还有专门针对某些知识点撰写的指南性质文档可供查阅。值得注意的是,社区内也存在许多由用户自发组织的研究小组和讨论区,这些都是获取一手资料以及与其他爱好者交流的好去处。 ```python import requests from bs4 import BeautifulSoup def fetch_graph_problems(): url = 'https://codeforces.com/problemset' params = {'tags': 'graphs'} response = requests.get(url, params=params) soup = BeautifulSoup(response.text, 'html.parser') problems = [] for row in soup.select('.problems tr'): columns = row.find_all('td') if len(columns) >= 2: problem_name = columns[0].text.strip() link = f"https://codeforces.com{columns[0].a['href']}" difficulty = columns[-1].span.text.strip() if columns[-1].span else None problems.append({ 'name': problem_name, 'link': link, 'difficulty': difficulty }) return problems[:5] fetch_graph_problems() ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值