作者热门文章
- VisualStudio2022插件的安装及使用-编程手把手系列文章
- pprof-在现网场景怎么用
- C#实现的下拉多选框,下拉多选树,多级节点
- 【学习笔记】基础数据结构:猫树
重温Tarjan, 网上看了许多博客感觉都讲的不清楚. 故传上来自己的笔记, 希望帮到大家. 。
提到的一些概念可以参考 oi wiki, 代码也是 oi wiki 的, 因为我不认为我能写出比大佬更好的代码了. 。
强连通分量: 有向图的最大强连通子图 ( 有向图中任意两点可达 ) 。
Tarjan 。
对每个结点维护
dfn[x]: 当前节点的 dfs 序. 。
low[x]: x 向下搜索能到达的最小 dfs 序. 。
更新 low
v 未被访问过: 初始 low[v] = dfn[v].v 入栈. 回溯时用 low[v] 更新它的 fa 的 low[ ]. 。
v 被访问过, 且还在栈中: 用 dfs[v] 更新 fa 的 low. 。
v 被访问过, 不在栈中: 说明这是一个 fa 到 v 的单向访问, 跳过. 。
获取答案
能让 dfn[x] > low[x], 只有当 X 的子树中某个节点 C 有\(\begin {cases}1.一条横向边连接到一棵已遍历过的子树~A\\2.一条返祖边连接到~X~的祖先~xfa \end{cases}\) . 。
dfn[xfa] = low[xfa]
.此时栈中进来过三类节点
故, 回溯时若节点符合 dfn[x] = low[x], 说明当前节点是它所属连通块的最小节点. 栈里它之上所有点都是一个强连通块. 。
代码
const int Maxn = 1e5 + 10;
int dfn[Maxn], low[Maxn], dfncnt, s[Maxn], in_stack[Maxn], tp;
int scc[Maxn], sc; // 结点 i 所在 SCC 的编号
int sz[Maxn]; // 强连通 i 的大小
void tarjan(int u) {
low[u] = dfn[u] = ++dfncnt, s[++tp] = u, in_stack[u] = 1;
for (int i = head[u]; i; i = eg[i].nex) {
const int &v = eg[i].to;
if (!dfn[v]) {
tarjan(v);
low[u] = min(low[u], low[v]);
} else if (in_stack[v]) {
low[u] = min(low[u], dfn[v]);
}
}
if (dfn[u] == low[u]) {
++sc;
while (s[tp] != u) {
scc[s[tp]] = sc;
sz[sc]++;
in_stack[s[tp]] = 0;
--tp;
}
scc[s[tp]] = sc;
sz[sc]++;
in_stack[s[tp]] = 0;
--tp;
}
}
最后此篇关于Tarjan求有向图的强连通分量的文章就讲到这里了,如果你想了解更多关于Tarjan求有向图的强连通分量的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我是一名优秀的程序员,十分优秀!