templete_tarjan求scc

tarjan这个人很神奇呀,发表的算法非常的多,这个求有向图的强连通量,也是人家发明的。这个算法可以求出图中的强连通分量数目,和分别每个顶点所对应的强连通分量的编号,十分的好用呀,结合上其他的算法,就可以将图论的连通性展现的淋漓尽致呀。


直接贴出代码吧:

#include <stdio.h>
#include <string.h>
#include <iostream>
#include <vector>
#include <stack>
#include <string>

/*
Õâ¾ÍÊǷdz£¾­µäµÄtarjanËã·¨£¬Õâ¸öËã·¨µÄÖ÷Òª×÷ÓþÍÊÇÇó³öÓÐÏòͼµÄÇ¿Á¬Í¨ÊýÁ¿£¬
²¢ÇÒ½«Ã¿¸öÇ¿Á¬Í¨·ÖÁ¿·Ö±ð±êºÅ£¬ÄÇôֻҪ±êºÃºÅÁËÖ®ºó£¬ÄǾͿÉÒ԰ɱêºÅÊÇÒ»ÑùµÄµãÈ«²¿
Ëù³ÆÒ»¸öµãÁË£¬ÒòΪһ¸öÇ¿Á¬Í¨Í¼ÉϵÄËùÓе㶼ÊÇ¿ÉÒÔ»¥Ïൽ´ïµÄ£¬¿ÉÒÔËõ³ÉÒ»¸öµã£¬
Èç¹ûÒªÇóËùÓеÄÇ¿Á¬Í¨·ÖÁ¿£¬Á¬³ÉÒ»¸ö£¬ÕûÌåµÄÇ¿Á¬Í¨·ÖÁ¿£¬¾ÍҪͳ¼Æÿ¸öËõµãºóµÄµãµÄÈë¶ÈΪ0µÄµã£¬
³ö¶ÈΪ0µÄµã¡£È»ºó±È½ÏÁ½ÕߵĴóС£¬´óµÄ¾ÍÊÇÒªÌí¼ÓµÄ±ß¡£ÕâÊÇÒòΪÿ¸öµãÈôÏëÁªÍ¨ËùÓеĵ㣬
±ØÐëÊÇÒªÓиö³ö¶ÈºÍÈë¶ÈµÄ¡£ 
*/

using namespace std;

const int MAXN = 100 + 11;

int N, M;

vector <int> G[MAXN];

int pre[MAXN], lowlink[MAXN], sccno[MAXN], dfs_clock, scc_cnt;

stack <int> S;

void init()
{
	for (int i = 0; i <= N; i++)
	{
		G[i].clear();
	}
}

void DFS(int u)
{
	pre[u] = lowlink[u] = ++dfs_clock;
	S.push(u);
	for (int i = 0; i < G[u].size(); i++)
	{
		int v = G[u][i];
		if (!pre[v])
		{
			DFS(v);
			lowlink[u] = min(lowlink[u], lowlink[v]); //Çó³ö´Ë½ÚµãÒÔϵÄ×Ó½ÚµãµÄ×îСʱ¼ä´Á¡£ 
		}
		else if (!sccno[v])//Èç¹ûÕâ¸öµã²»ÊôÓÚÈκÎÁ¬Í¨·ÖÁ¿¡£ 
		{
			lowlink[u] = min(lowlink[u], pre[v]);
		}
	}
	if (lowlink[u] == pre[u])
	{
		scc_cnt++;
		for (; ;)
		{
			int x = S.top();
			S.pop();
			sccno[x] = scc_cnt;
			if (x == u)
			{
				break;
			}
		}
	}
}

void find_scc()
{
	dfs_clock = scc_cnt = 0;
	memset(sccno, 0, sizeof(sccno)); //sccnoÊÇÓÃÀ´´æÿ¸öµã¶ÔÓ¦µÄÁ¬Í¨·ÖÁ¿µÄ±êºÅµÄ¡£ 
	memset(pre, 0, sizeof(pre)); //pre»¹ÊÇÀϹæ¾Ø£¬ÓÃÀ´±ê¼ÇÊÇ·ñ·ÃÎÊ£¬ºÍµ±×öʱ¼ä´Á¡£ 
	for (int i = 1; i <= N; i++)
	{
		if (!pre[i])
		{
			DFS(i);
		}
	}
}

int main()
{
	while (scanf("%d%d", &N, &M) != EOF)
	{
		init();
		int a, b;
		for (int i = 0; i < M; i++)
		{
			scanf("%d%d", &a, &b);
			G[a].push_back(b);
		}
		find_scc();
		printf("the scc_cnt = %d\n", scc_cnt);
	} 
	return 0;
} 

/*
6 8
1 2
2 3
2 4
2 5
5 2
5 6
3 6
6 3

*/


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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值