P2863The Cow Prom S(tarjan算法)

有一个 n 个点,m 条边的有向图,请求出这个图点数大于 1 的强联通分量个数。

#include<stdio.h>
#include<iostream>
#include<algorithm>
#include<cstring>
#include<vector>
#include<map>
#include<queue>
#include<stack>
#include<math.h>
#include<set>
using namespace std;
vector<int>p[10010];
stack<int>s;
bool book[10010], vis[10010];
int dfn[10010], low[10010], step = 1, f;//dfn[]记录到达的时间,low[]记录到达的最近的点
int a[10010], sum = 0;
void tarjan(int x)
{
	dfn[x] = low[x] = step++;
	s.push(x);//入栈
	book[x] = vis[x] = true;
	for (int i = 0; i < p[x].size(); i++)
	{
		int c = p[x][i];
		if (!dfn[c])
		{
			tarjan(c);
			low[x] = min(low[x], low[c]);//回溯
		}
		else
		{
			if (vis[c])//如果还在栈内,更新low[]
			{
				low[x] = min(low[x], dfn[c]);
			}
		}
	}
	if (low[x] == dfn[x])
	{
		sum++;//强连通分量的数量
		do
		{
			f = s.top();
			vis[f] = false;
			s.pop();
			a[sum]++;
		} while (x != f);//x及前面的点都出栈
	}
}
int main()
{
	int n, m;
	cin >> n >> m;
	for (int i = 0; i < m; i++)
	{
		int x, y;
		cin >> x >> y;
		p[x].push_back(y);
	}
	for (int i = 1; i <= n; i++)
	{
		if (!book[i])
		{
			tarjan(i);//保证所有点都有访问到
		}
	}
	int ans = 0;
	for (int i = 1; i <= sum; i++)
	{
		if (a[i] > 1)
		{
			ans++;
		}
	}
	cout << ans;
	return 0;
}

样例输入
5 4
2 4
3 5
1 2
4 1
样例输出
1

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值