返回顶部
首页 > 资讯 > 前端开发 > JavaScript >怎么使用Tarjan算法求解强连通分量
  • 727
分享到

怎么使用Tarjan算法求解强连通分量

2024-04-02 19:04:59 727人浏览 泡泡鱼
摘要

本篇内容介绍了“怎么使用Tarjan算法求解强连通分量”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!算法框

本篇内容介绍了“怎么使用Tarjan算法求解强连通分量”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!

算法框架

我们来思考一个问题,对于强连通分量分解的算法来说,它的核心原理是什么?

如果你看过我们之前的文章,那么这个问题对你来说应该不难回答。既然是强连通分量,意味着分量当中每个点都可以互相连通。所以我们很容易可以想到,我们可以从一个点出发,找到一个回路让它再回到起点。这样途中经过的点就都是强连通分量的一部分。

但是这样会有一个问题,就是需要保证强连通分量当中的每个点都被遍历到,不能有遗漏。针对这个问题我们也可以想到解法,比如可以用搜索算法去搜索它所有能够达到的点和所有的路径。但是这样一来,我们又会遇到另外一个问题。这个问题就是强连通分量之间的连通问题。

我们来看个例子:

怎么使用Tarjan算法求解强连通分量

在上面这张图当中如果我们从点1出发,我们可以达到图中的每一个点。但是我们会发现1,2,3是一个强连通分量,4,5,6是另外一个。当我们寻找1所在的强连通分量的时候,很有可能会把4,5,6这三个点也带进来。但问题是它们是自成分量的,并不应该算在1的强连通分量当中。

我们整理一下上面的分析和思路可以发现强连通分量分解这个算法的核心其实就是解决这两个问题,就是完备性问题。完备意味着不能遗漏也不能冗余和错误,我们想明白核心问题所在之后就很容易搭建起思维框架,接下来我们再来看算法的描述会容易理解得多。

算法细节

Tarjan算法的第一个机制是时间戳,也就是在遍历的时候对每一个遍历到的点打上一个值。这个值表示这是第几个遍历的元素。

这个应该很好理解,我们只需要维护一个全局的变量,在遍历的时候去让它自增就可以了。我们来写下python代码给大家演示一下:

stamp = 0 stamp_dict = {}def dfs(u):     stamp_dict[u] = stamp    stamp += 1     for v in Graph[u]:         dfs(v)

通过时间戳我们可以知道每个点被访问的顺序,这个顺序是正向顺序。举个例子,比如说假设u和v两个点,u的时间戳比v小。那么它们之间的关系只有两种可能,第一种是u能够连通到v,说明从u到v的链路可以走通。第二种是u不能连通到v,这种情况不论反向的从v到u能否连通都不具有讨论意义,因为它们一定不能互相连通。

所以我们想要找到连通的通路还需要找到反向的路径,在Kosaraju算法当中我们是通过反向图来实现的。在Tarjan当中则采取了另外一种方法。因为我们已经知道各个点的时间戳了,我们完全可以通过时间戳来寻找反向的路径。什么意思呢?其实很简单,当我们在遍历u的时候如果遇到了一个比u时间戳更小的v,那么说明就存在一条反向的路径从u通向v。如果v这时候还没有出栈,意味着v是u的上游的话,那么也就说明存在一条路径从v通向u。这样就说明了u和v可以互相连通。

既然找到了一对互相连通的u和v,那么我们需要把它们记录下来。但问题是我们怎么知道记录到什么时候为止呢?这个边界在哪里?Tarjan算法设计了另外一个巧妙的机制解决了这个问题。

这个机制就是low机制,low[u]表示u这个点能够连通到的所有的点的时间戳的最小值。时间戳越小说明在搜索树当中的位置越高,也可以理解成u能够连通到的处在搜索树中最高的点。那么很明显了,这个点就是u这个点所在强连通分量所在搜索树某一棵子树的树根。

这里可能有一点点绕,我们再来看张图:

怎么使用Tarjan算法求解强连通分量

图中节点所在的序号就是递归遍历的时间戳,我们可以发现对于图上的每个点来说它们的low值都是1。很明显1这个点在搜索树当中是2,3,4这三个点的祖先。也就是说这一个强连通分量的遍历是从1这个点开始的。当1这个点出栈的时候,意味着以1位树根的子树已经遍历完了,所有可能存在的强连通分量也都已经找完了。

这就带来了另外一个问题,我们假设当前点是u,我们如何知道u这个点是否是图中1这样的树根呢?有没有什么办法可以标记出来呢?

当然是有的,这样的点有一个特性就是它们的时间戳等于它们的low。所以我们可以用一个数组维护找到的强连通分量,当这些强连通分量能够遍历到的树根出栈的时候,把数组清空。

我们把上面的逻辑整理一下就可以写出代码来了:

scc = [] stack = [] def tarjan(u):    dfn[u], low[u] = stamp, stamp    stamp += 1  stack.append(u)         for v in Graph[u]:         if not dfn[v]:             tarjan(v)            low[u] = min(low[u], low[v])        elif v in stack:          low[u] = min(low[u], dfn[v])       if dfn[u] == low[u]:         cur = []        # 栈中u之后的元素是一个完整的强连通分量        while True:             cur.append(stack[-1])             stack.pop()             if cur[-1] == u:                 break         scc.append(cur)

最后,我们来看一下之前讲过的经典例子:

怎么使用Tarjan算法求解强连通分量

首先我们从1点开始,一直深搜到6结束,当遍历到6的时候,DFN[6]=4,low[6]=4,当6出栈时满足条件,6独立称为一个强连通分量。

怎么使用Tarjan算法求解强连通分量

同理,当5退出的时候也同样满足条件,我们得到了第二个强连通分量。

怎么使用Tarjan算法求解强连通分量

接着我们回溯到节点3,节点3还可以遍历到节点4,4又可以连向1。由于1点已经在栈中,所以不会继续递归1点,只会更新low[4] =  1,同样当4退出的时候又会更新3,使得low[3] = 1。

怎么使用Tarjan算法求解强连通分量

最后我们返回节点1,通过节点1遍历到节点2。2能连通的4点已经在栈中,并且DFN[4] >  DFN[2],所以并不会更新2点。再次回到1点之后,1点没有其他点可以连通,退出。退出的时候发现low[1] =  DFN[1],此时栈中剩下的4个元素全部都是强连通分量。

怎么使用Tarjan算法求解强连通分量

“怎么使用Tarjan算法求解强连通分量”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注编程网网站,小编将为大家输出更多高质量的实用文章!

--结束END--

本文标题: 怎么使用Tarjan算法求解强连通分量

本文链接: https://lsjlt.com/news/84586.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • 怎么使用Tarjan算法求解强连通分量
    本篇内容介绍了“怎么使用Tarjan算法求解强连通分量”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!算法框...
    99+
    2024-04-02
  • Android岛屿数量算法怎么使用
    这篇文章主要介绍“Android岛屿数量算法怎么使用”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Android岛屿数量算法怎么使用”文章能帮助大家解决问题。岛屿数量之前接触过一个算法,比较有意思,...
    99+
    2023-07-05
  • 怎么使用python递归算法求n的阶乘
    你可以使用下面的代码来使用递归算法求n的阶乘:```pythondef factorial(n):if n == 0 or n ==...
    99+
    2023-08-09
    python
  • PHP怎么用回溯算法求解子集问题
    本篇内容介绍了“PHP怎么用回溯算法求解子集问题”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!回溯算法实际上一个类似枚举的搜索尝试过程,主要...
    99+
    2023-06-20
  • python图像分割算法怎么使用
    Python中常用的图像分割算法有基于阈值的分割算法、基于边缘的分割算法和基于区域的分割算法。以下是使用这些算法的示例代码:1. 基...
    99+
    2023-10-18
    python
  • 怎么使用java递归算法求最大公约数
    要使用递归算法求最大公约数,可以按照以下步骤进行:1. 创建一个名为"gcd"的递归函数,接受两个整数参数a和b,并返回它们的最大公...
    99+
    2023-08-09
    java
  • python加密解密算法怎么使用
    Python提供了多种加密解密算法的库,比如`hashlib`、`hmac`、`base64`、`cryptography`等。下面...
    99+
    2023-09-17
    python
  • 怎么使用PHP实现分布算法之一致性哈希算法
    这篇文章主要介绍怎么使用PHP实现分布算法之一致性哈希算法,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!传统算法缺陷对于服务器分布,我们要考虑的东西有如下三点:数据平均分布,查找定位准确,降低宕机影响。传统算法一般是...
    99+
    2023-06-15
  • 轻量应用服务器怎么连接电脑使用方法
    轻量应用服务器是一种用于处理小型、轻量级应用程序的服务器,通常由小型团队管理和维护。以下是几种连接轻量应用服务器的方法: Python内置连接:使用Python的内置pip库可以连接到轻量应用服务器上,例如使用pynet pip就可以使...
    99+
    2023-10-26
    使用方法 服务器 电脑
  • ssl连接错误无法使用怎么解决
    如果您遇到 SSL 连接错误,可能是由于以下原因导致的:1. 证书问题:如果您的网站使用自签名证书或过期证书,则可能会导致 SSL ...
    99+
    2023-06-03
    ssl连接 ssl
  • 怎么使用pytorch进行张量计算、自动求导和神经网络构建功能
    本文小编为大家详细介绍“怎么使用pytorch进行张量计算、自动求导和神经网络构建功能”,内容详细,步骤清晰,细节处理妥当,希望这篇“怎么使用pytorch进行张量计算、自动求导和神经网络构建功能”文章能帮助大家解决疑惑,下面跟着小编的思路...
    99+
    2023-07-06
  • 驱动程序无法通过使用SSL加密与SQL Server建立安全连接怎么解决
    这篇文章主要介绍“驱动程序无法通过使用SSL加密与SQL Server建立安全连接怎么解决”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“驱动程序无法通过使用SSL加密与SQL S...
    99+
    2023-07-05
  • OpenCV-Python怎么使用分水岭算法实现图像分割与提取功能
    小编给大家分享一下OpenCV-Python怎么使用分水岭算法实现图像分割与提取功能,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!随着当今世界的发展,计算机视觉技...
    99+
    2023-06-15
  • 怎么使用Python贪心算法解决集合覆盖问题
    本文小编为大家详细介绍“怎么使用Python贪心算法解决集合覆盖问题”,内容详细,步骤清晰,细节处理妥当,希望这篇“怎么使用Python贪心算法解决集合覆盖问题”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。在《算...
    99+
    2023-06-04
  • 怎么使用Java方法调用解析静态分派和动态分派
    这篇“怎么使用Java方法调用解析静态分派和动态分派”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“怎么使用Java方法调用解...
    99+
    2023-07-02
  • Retrofit网络请求框架之注解解析和动态代理方法怎么使用
    本篇内容介绍了“Retrofit网络请求框架之注解解析和动态代理方法怎么使用”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!Retrofit是...
    99+
    2023-07-05
  • Java8(291)之后禁用了TLS1.1使JDBC无法用SSL连接SqlServer2008怎么解决
    这篇文章主要介绍“Java8(291)之后禁用了TLS1.1使JDBC无法用SSL连接SqlServer2008怎么解决”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Java8(291)之后禁用了T...
    99+
    2023-07-05
  • 怎么使用SAP SAT事务码对通过浏览器启动的应用的性能测量和分析方式
    怎么使用SAP SAT事务码对通过浏览器启动的应用的性能测量和分析方式,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。三个产品都有登录语言的选择:CRMC4C:Hybris:...
    99+
    2023-06-04
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作