我们可以使用 tarjan 来求取一张有向图中强连通分量的个数,以及每个强连通分量包含的点的编号.
强连通#
在一张有向图中,如果存在由一些点组成的集合,满足这个集合中的点都可以两两连通,那么称这些点和边为原图的强连通子图.
如果对于一个强连通子图 ,不存在任意一个其他点在加入 后,能够仍然保持强连通子图的性质,那么称 为原图的强连通分量.通俗来说,强连通分量就是极大的强连通图.
DFS 生成树#
在有向图上,以任意一个节点为根节点进行一次 DFS,我们可以使用DFS 生成树记录其过程.对于一个节点 A,其子节点 B 满足没有在图上出现过、且有一条边从 A 连接到 B.
DFS 生成树可以辅助将图转换为树 + 边.
示例:
flowchart LR
A((A)) --> B((B))
A --> D((D))
D --> A
B --> C((C))
D --> E((E))
E --> F((F))
E --> H((H))
F --> G
B --> E
E --> G((G))
对于上面这条边,如果以 A 为根节点建 DFS 生成树,一种是这样的:
flowchart TD
A((A)) --> B((B))
B --> C((C))
%% cross
B -.-> E((E))
E --> H((H))
E --> F((F))
F --> G((G))
%% forward
E -.-> G
A --> D((D))
D --> E
%% back
D -.-> A
linkStyle 2 stroke:blue
linkStyle 6 stroke:green
linkStyle 9 stroke:red
在这张图中,有四种边:
- 树边:原有的,构成生成树的边 除此以外,还有三种额外的边(假定这些边是从节点 连接到节点 的):
- 反祖边: 是 的祖先节点.
- 前向边: 不是 的父节点,但是是 的祖先节点.
- 横叉边:不是树边、反祖边和前向边的边.
不难发现,每一个强连通分量都至少有一条反祖边.
如何在 DFS 的同时判断一条边是什么边?
我们需要定义一些数据:
bool in_stack[i]表示点 是否在当前 DFS 的路径上.bool vis[i]表示点 是否被访问过.
in_stack 成立 | in_stack 不成立 | |
|---|---|---|
vis[i] 成立 | 反祖边 | 前向边或横叉边 |
vis[i] 不成立 | 不可能 | 树边 |
Tarjan 求强连通分量#
Tarjan 只需要进行一次 DFS 搜索就能办到.
为了解决问题,我们定义了以下数据:
bool in_stack[i]表示点 是否在当前 DFS 的路径上.stack<int> s当前强连通分量中的节点编号.int dfn[i]第 个点先序遍历到达时间.int low[i]第 个点通过当前子树内返祖边(即不经过其父节点)回到的最小时间.- 无需定义
vis[i],应为vis[i]等效于dfn[i] == 0.
不难发现:
- 祖先
dfn一定小于当前dfn. - 如果一个点 能够到达一个点 ,点 u 的
dfn大于点 v 的dfn,那么这样的一组构成了一个强连通分量
对于到达一个新的节点 ,我们会进行以下操作:
- 初始化操作
- 初始化
dfn[u] - 初始化
low[u] = dfn[u] - 将当前节点入栈
s,并且标记in_stack.
- 初始化
- 遍历所有起点为 ,终点为 的边
- 如果这条边是反祖边,将
low[u]进行更新. - 如果这条边是树边,DFS 搜索节点 ,然后将
low[u]进行更新. - 如果这条边是横叉边或前向边,那么说明 已搜索完毕,其所在连通分量已被处理,所以不用对其做操作.
- 如果这条边是反祖边,将
- 检查当前节点是不是一个新的强连通分量的根节点(
low[u] == dfn[u])- 如果是,那么当前
s中存储的就是一个强连通分量.清空s,初始化in_stack来处理下一个强连通分量.
- 如果是,那么当前
时间复杂度分析#
- 初始化操作:
- 遍历边操作:
- 检查节点是不是强连通分量的根节点:
综合一下,时间复杂度
标程#
下面这个程序求解了图中的强连通分量数量和每一个强连通分量的组成部分.
#include <bits/stdc++.h>
using namespace std;
const int N = 1e4 + 100;
int m, n, ans;
vector<int> e[N];
//bool in_stack[i]表示点i是否在当前DFS的路径上.
//stack<int> s当前强连通分量中的节点编号.
//int dfn[i]第i个点先序遍历到达时间.
//int low[i]第i个点通过当前子树内返祖边回到的最小时间.
int dfn[N], low[N], cnt;
bool in_stack[N];
stack<int> s;
void tarjan(int u)
{
if (dfn[u] != 0)
return;
// 初始化
dfn[u] = ++cnt;
low[u] = dfn[u];
in_stack[u] = true;
s.push(u);
// 搜索边
for (auto v : e[u])
{
// 如果是返祖边
if (in_stack[v])
{
// 那么刷新low[u]
low[u] = min(low[u], low[v]);
}
// 如果是树边
else if (!dfn[v])
{
// 递归处理
tarjan(v);
low[u] = min(low[u], low[v]);
}
// 如果是横插边或前向边
else
{
// 节点v已经被处理或将被处理
// 什么也不用做
}
}
// 检查现在的节点是不是强连通分量的根节点.
if (low[u] == dfn[u]) // 如果说这个节点最小只能到达自己
{
// 那么这个节点就是强连通分量的根节点.
ans++;
vector<int> v;
while (s.top() != u)
{
v.push_back(s.top());
in_stack[s.top()] = false;
s.pop();
}
v.push_back(s.top());
in_stack[s.top()] = false;
s.pop();
cout << v.size() << " ";
for (auto it : v)
{
cout << it << " ";
}
cout << endl;
}
}
signed main()
{
#ifndef ONLINE_JUDGE
freopen(R"(E:\code\C++\sandbox\sandbox\in.txt)", "r", stdin);
#endif
cin >> n >> m;
for (int i = 1; i <= m; i++)
{
int u, v;
cin >> u >> v;
e[u].push_back(v);
}
for (int i = 1; i <= n; i++)
{
tarjan(i);
}
cout << ans << endl;
return 0;
}cpp