每日一题 2019/4/9

今天学一下tarjan求强联通分量,继续整理模板

 

好文:https://www.sohu.com/a/245954819_100201031

模板:

//输出所有的强联通分量
struct node {
	int v, nxt;
}edge[maxm];

int dfn[maxn], low[maxn];
int st[maxn], heads[maxn];
bool vis[maxn];//vis标记是否在栈里
int cnt, tot, index;

void add(int x, int y){
	edge[++cnt].nxt = heads[x];
	edge[cnt].v = y;
	heads[x] = cnt;
}

void tarjan(int x){
	dfn[x] = low[x] = ++tot;
	st[++index] = x;
	vis[x] = 1;
	for(int i = heads[x]; i != -1; i = edge[i].nxt){
		if(!dfn[edge[i].v]){
			tarjan(edge[i].v);
			low[x] = min(low[x],low[edge[i].v]);
		}
		else if(vis[edge[i].v]){//如果访问过,并且还在栈里。
			low[x] = min(low[x], dfn[edge[i].v]);//比较谁是谁的儿子/父亲。就是链接对应关系
		}
	}
	if(low[x] == dfn[x]){//强连通分量子树里的最小根
		do{
			printf("%d ", st[index]);
			vis[st[index]] = 0;
			index--;
		}while(x != st[index+1]);
		puts("");
	}
}

int main()
{
	cnt = index = 0;
	memset(heads, -1, sizeof(heads));
	int n, m;
	scanf("%d %d", &n, &m);
	rep(i, 1, m){
		int x, y; scanf("%d %d", &x, &y);
		add(x, y);//x->y
	}
	rep(i, 1, n) if(!dfn[i]) tarjan(i);
	return 0;
}

 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值