Tarjan‘s Algorithm Practice

Tarjan’s Algorithm——求强连通分量的线性时间的算法

  • 如果有两个vertex可以互相到达,那么则称两个顶点强连通

  • 若graph中任意两个顶点都可以互相到达,那么则称这个图为一个强连通图
    其中的一个子图称为强连通分量

  • 基于对图的dfs,每个强连通分量是搜索树中的一颗子树

  • 维护两个数组dfn[N],low[N],dfn[x]表示顶点x的时间戳,low[x]表示x或x的子树能够回溯的栈中最小的次序

模板题1

题目描述
给出一个 n 个点,m 条边的无向图,求图的割点。

输入格式
第一行输入两个正整数 n,m。

下面m 行每行输入两个正整数 x,y表示 x 到 y 有一条边。

输出格式
第一行输出割点个数。

第二行按照节点编号从小到大输出节点,用空格隔开。

输入输出样例

6 7
1 2
1 3
1 4
2 5
3 5
4 5
5 6
1
5

Code
Attention:“index” is a keyword int C++

#include<bits/stdc++.h>
using namespace std;

int n, m;
const int N = 2e4 + 5;
const int M = 1e5 + 5;
struct Edge {
	int end, next;
}edge[2*M];
int head[N], dfn[N], low[N];
bool vis[N];
int cnt;
void Tarjan(int x,int y) {
	dfn[x] = low[x] = ++cnt;
	int child = 0;
	for (int i = head[x]; i; i = edge[i].next) {
		int end = edge[i].end;
		if (!dfn[end]) {
			Tarjan(end, y);
			low[x] = min(low[x], low[end]);
			if (low[end] >= dfn[x] && x != y) {
				vis[x] = true;
			}
			if (x == y)child++;
		}
		/*else*/ low[x] = min(low[x], dfn[end]);
	}
	if (child >= 2 && x == y)vis[x] = true;
}
int main() {
	ios::sync_with_stdio(false);
	cin.tie(0); cout.tie(0);
	cin >> n >> m;
	int t = 0;
	for (int i = 1; i <= m; i++) {
		int u, v;
		cin >> u >> v;
		++t;
		edge[t].end = v;
		edge[t].next = head[u];
		head[u] = t;
		++t;
		edge[t].end = u;
		edge[t].next = head[v];
		head[v] = t;
	}
	for (int i = 1; i <= n; i++) {
		if (!dfn[i]) {
			Tarjan(i, i);
		}
	}
	int ans = 0;
	for (int i = 1; i <= n; i++) {
		if (vis[i])ans++;
	}
	cout << ans << endl;
	for (int i = 1; i <= n; i++) {
		if (vis[i])cout << i << " ";
	}
	return 0;
}

模板题2

题目描述
有n个城市,中间有单向道路连接,消息会沿着道路扩散,现在给出n个城市及其之间的道路,问至少需要在几个城市发布消息才能让这所有n个城市都得到消息。

输入格式
第一行两个整数n,m表示n个城市,m条单向道路。

以下m行,每行两个整数b,e表示有一条从b到e的道路,道路可以重复或存在自环。

输出格式
一行一个整数,表示至少要在几个城市中发布消息。

输入输出样例

5 4
1 2
2 1
2 3
5 1
2

Code

#define _CRT_SECURE_NO_WARNINGS
#include<bits/stdc++.h>
using namespace std;

int n, m;
const int N = 1e5 + 5;
const int M = 5e5 + 5;
struct Edge {
	int end, next;
}edge[M];
int head[N], dfn[N], low[N], Stack[N], belong[N], indeg[N];
bool vis[N];
int cnt, ind, sum;
void Tarjan(int x) {
	dfn[x] = low[x] = ++cnt;
	Stack[++ind] = x;
	vis[x] = true;
	for (int i = head[x]; i; i = edge[i].next) {
		int end = edge[i].end;
		if (!dfn[end]) {
			Tarjan(end);
			low[x] = min(low[x], low[end]);
		}
		else if(vis[end]){
			low[x] = min(low[x], dfn[end]);
		}
	}
	if (dfn[x] == low[x]) {
		++sum;
		do {
			int tmp = Stack[ind];
			--ind;
			vis[tmp] = false;
			belong[tmp] = sum;
		} while (x != Stack[ind + 1]);
	}
}
int main() {
	ios::sync_with_stdio(false);
	cin.tie(0); cout.tie(0);
	cin >> n >> m;
	for (int i = 1; i <= m; i++) {
		int u, v;
		cin >> u >> v;
		if(u == v)continue;//避免成环
		edge[i].end = v;
		edge[i].next = head[u];
		head[u] = i;
	}
	for (int i = 1; i <= n; i++) {
		if (!dfn[i])Tarjan(i);
	}
	for (int i = 1; i <= n; i++) {
		for (int j = head[i]; j; j = edge[j].next) {
			int end = edge[j].end;
			if (belong[end] != belong[i]) {
				indeg[belong[end]]++;
			}
		}
	}
	int ans = 0;
	for (int i = 1; i <= sum; i++) {
		if (!indeg[i])ans++;
	}
	cout << ans << endl;
	return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值