首页 > 编程语言 >「学习笔记」tarjan 算法与强连通分量

「学习笔记」tarjan 算法与强连通分量

时间:2023-04-26 15:26:17浏览次数:37  
标签:back tarjan 连通 scc st 算法 dfn low

强连通的定义是:有向图 G 强连通是指,G 中任意两个结点连通。
强连通分量(Strongly Connected Components,SCC)的定义是:极大的强连通子图。
说简单一点就是,环内的点都在一个强连通分量里,单独一个点也算是强连通分量(自己可以到达自己)。

变量

int tim, sc;
int dfn[N], low[N], scc[N];
vector<int> son[N], st;

tim: 时间戳计数器;
sc: 强连通分量数量;
dfn: 时间戳;
low: 通过一条返祖边向上能到达的最靠上的点;
scc: 所属的强联通分量;
son[N]: 存图;
st: 栈;

过程

image
在这张图上
红色的边是返祖边,蓝色的边是横叉边,绿色的边是前向边,黑色的边是树边。
我们从一个点开始 dfs,给每一个节点打上时间戳,在没有遇到返祖边之前,low 值为这个点自己,该节点入栈,在栈中的节点都是还没有确定所属强连通分量的节点,出栈的节点是已经找到强连通分量的点

dfn[u] = low[u] = ++ tim;
st.push_back(u);

若搜到的下一个点还没有被打过时间戳,则说明这个点从没有被搜到过,先搜索,随后更新 low 值;倘若它被打了时间戳,但仍在栈中,说明当前这条边是返祖边,则要更新 low 值。

for (int v : son[u]) {
	if (!dfn[v]) {
		tarjan(v);
		low[u] = min(low[u], low[v]);
	}
	else if (!scc[v]) {
		low[u] = min(low[u], dfn[v]);
	}
}

这里有个小细节,在 !dfn[v] 的条件下,low[u] = min(low[u], low[v]);,而在 !scc[v] 的条件下,low[u] = min(low[u], dfn[v]);
这是因为 !dfn[v] 的条件下,我们走的是树边或前向边,节点位置一定在该节点之下,子节点 \(v\) 能到达的最靠上的位置就是 \(u\) 能到达的最靠上的位置;而 !scc[v] 的条件下,我们走的是返祖边,节点位置在该节点上边,所以 low[u] = min(low[u], dfn[v]);,而不是 low[u] = min(low[u], low[v]);,因为 \(v\) 不是 \(u\) 的子节点,且 \(u\) 点不一定能通过一条返祖边到达 low[v]
如果一个节点的 dfn 等于 low,那么,说明这个点走返祖边能到达的最靠上的点就是它自己,就树上来说,它就是这个强连通分量中最靠上的点,截止到目前为止,在这个节点之后入栈的点就都是这个强连通分量的点,自己可以画画图看看。

if (dfn[u] == low[u]) {
	scc[u] = ++ sc;
	while (st.back() != u) {
		scc[st.back()] = sc;
		st.pop_back();
	}
	st.pop_back();
}

完整代码:

void tarjan(int u) {
	dfn[u] = low[u] = ++ tim;
	st.push_back(u);
	for (int v : son[u]) {
		if (!dfn[v]) {
			tarjan(v);
			low[u] = min(low[u], low[v]);
		}
		else if (!scc[v]) {
			low[u] = min(low[u], dfn[v]);
		}
	}
	if (dfn[u] == low[u]) {
		scc[u] = ++ sc;
		while (st.back() != u) {
			scc[st.back()] = sc;
			st.pop_back();
		}
		st.pop_back();
	}
}

应用

  • 缩点
    将每一个强连通分量缩成一个点,建出一个新图,这样原本有环的图就变成了一个 DAG(有向无环图),可以用来拓扑排序和 DP,缩点要维护原来环中的点的信息,如点权之类的。
for (int i = 1; i <= m; ++ i) {
	int u = e[i].u, v = e[i].v;
	if (lt[u] != lt[v]) {
		nson[lt[u]].push_back(lt[v]);
	}
}
  • 2-SAT 问题
    判断是否有环来判断条件是否冲突。

标签:back,tarjan,连通,scc,st,算法,dfn,low
From: https://www.cnblogs.com/yifan0305/p/17356068.html

相关文章

  • 单窗算法的地表温度反演:谷歌地球引擎GEE代码
      本文介绍在GEE中基于Landsat遥感影像实现地表温度(LST)单窗算法反演的代码。1背景知识  基于遥感数据的地表温度(LST)反演目前得到了广泛的应用,尤其是面向大尺度、长时间范围的温度数据需求,遥感方法更是可以凸显其优势。目前,基于各类遥感数据源的地表温度反演方法不断得以改......
  • 机器学习(七):梯度下降解决分类问题——perceptron感知机算法与SVM支持向量机算法进行二
    实验2感知机算法与支持向量机算法一、预备知识1.感知机算法二、实验目的掌握感知机算法的原理及设计;掌握利用感知机算法解决分类问题。三、实验内容设计感知机算法求解,设计SVM算法求解(可调用函数库),请找出支持向量和决策超平面。四、操作方法和实验步骤1.......
  • Erdős–Rényi 随机图的连通性
    对于给定的\(n\)个顶点,对于任意一个点对,以\(p\)的概率连边,这样得到的一个无向简单图上的概率分布,称为Erdős–Rényi随机图模型.那么,\(p\)有多大的时候,得到的图将会有很大概率连通呢?Erdős和Rényi给出了如下结果:对于\(p=(\logn+c)/n\),记事......
  • pid算法函数实现,c语言版
     #include<stdio.h>floatpid(floatsetpoint,floatprocess_variable,floatkp,floatki,floatkd,floatdt,float*integral,float*last_error){//Calculateerrorfloaterror=setpoint-process_variable;//Calculateintegral......
  • 滑动窗口算法实现分布式第三方请求限频
    一.业务背景 第三方服务接口存在频率调用限制(例如,1s5次,超过5次返回超出频率),己方服务存在并发处理的情况,为了保证服务的成功率,且达到第三方限制的最大吞吐量,故需要一个限频调用的算法二.实现思路常见限频算法一般有五种,漏桶算法、令牌桶算法、固定窗口算法,滑动窗口算法,漏斗算......
  • 基于ICP配准算法的三维点云数据的匹配仿真
    1.算法仿真效果matlab2022a仿真结果如下:       2.算法涉及理论知识概要       ICP算法能够使不同的坐标下的点云数据合并到同一个坐标系统中,首先是找到一个可用的变换,配准操作实际是要找到从坐标系1到坐标系2的一个刚性变换。ICP算法本质上是基于最小......
  • m基于LOC-PCA算法的人脸重建算法matlab仿真,给定人物侧脸实现正脸重建
    1.算法仿真效果matlab2022a仿真结果如下:       2.算法涉及理论知识概要      提出了一种有效的图像姿态合成方法。姿势合成用于预测在给定姿势的期望姿势处具有最小误差的面部图像。在许多情况下,这是经常需要的例如动画电影的制作、法医学和3D人脸几......
  • ARMA-EGARCH模型、集成预测算法对SPX实际波动率进行预测|附代码数据
    全文下载链接:http://tecdat.cn/?p=12174最近我们被客户要求撰写关于ARMA-EGARCH的研究报告,包括一些图形和统计输出。本文比较了几个时间序列模型,以预测SP500指数的每日实际波动率。基准是SPX日收益序列的ARMA-EGARCH模型。将其与GARCH模型进行比较 。最后,提出了集合预测算法......
  • 【优化指派】基于禁忌搜索算法求解指派优化问题(耗时最短)附Matlab代码
    ✅作者简介:热爱科研的Matlab仿真开发者,修心和技术同步精进,matlab项目合作可私信。......
  • 【LSSVM时序预测】基于被囊群算法优化最小支持向量机TSA-LSSVM实现交通流数据预测附Ma
    ✅作者简介:热爱科研的Matlab仿真开发者,修心和技术同步精进,matlab项目合作可私信。......