[算法模版]Tarjan爷爷的几种图论算法

前言

Tarjan爷爷发明了很多图论算法,这些图论算法有很多相似之处(其中一个就是我都不会)。这里会对这三种算法进行简单介绍。

定义

强连通(strongly connected): 在一个有向图\(G\)里,设两个点a, b 发现,由\(a\)有一条路可以走到\(b\),由\(b\)又有一条路可以走到\(a\),我们就叫这两个顶点(a,b)强连通。

强连通图: 如果 在一个有向图\(G\)中,每两个点都强连通,我们就叫这个图,强连通图。

分量:把一个向量分解成几个方向的向量的和,那些方向上的向量就叫做该向量(未分解前的向量)的分量。

强连通分量(strongly connected components/SCC):在一个有向图G中,有一个子图,这个子图每2个点都满足强连通,我们就叫这个子图叫做强连通分量。

比如说这个图,在这个图中呢,点1与点2互相都有路径到达对方,所以它们强连通。

而在这个有向图中,点1 2 3组成的这个子图,是整个有向图中的强连通分量。

dfn[i]:指第\(i\)个点的\(dfs\)序。

low[i]:指第\(i\)个点的子树内的所有点通过反祖边能走到的点的\(dfn\)的最小值。

先行结论

  • 在一个无向图上,跑一棵生成树。可以证明非树边只有反祖边,没有横叉边。

Tarjan算法求割点/割边(针对无向图)

割点

首先,一个比较显然的结论就是所有的叶子结点和根节点都不是割点。对于其他节点u来说,只要有至少一个儿子v满足low[u]>=dfn[v],就证明v若不通过他父亲就回不去。那这个点就是割点。根节点只需要判断是不是有两棵子树就好了。实现起来可以当作有两个点满足low[u]>=dfn[b]。因为因为根节点\(dfn\)为1,所以一定满足条件。如果两个根结点儿子不通过根节点就能联通,那么他们一定在一个子树。

void tarjan(int now,int ff){
low[now]=dfn[now]=++idx;
for(int i=head[now];i;i=side[i].next){
int v=side[i].v;
if(side[i].id==ff)continue;
if(!dfn[v]){
tarjan(v,side[i].id);
low[now]=min(low[now],low[v]);
if(low[v]>=dfn[now]){ans[now]++;}
}
else{
low[now]=min(dfn[v],low[now]);
}
}
return;
}
void output(){
for(int i=1;i<=n;i++){
if((ans[i]&&!root[i])||(ans[i]>=2&&root[i])){
cout<<i<<' ';//输出所有割点
}
}
}

割边

和割点基本一样。只需要把\(low[v]>=dfn[u]\)改成\(low[v]>dfn[u]\)即可。同时需要判断,不能再次走刚刚走过的边。

如果点u的至少一个儿子v满足low[v]>dfn[u]。就证明不通过这条边无法走到上面。所以这是条割边。

Tarjan算法求点双/边双(针对无向图)

边双

和楼下的强连通分量很像,唯一的区别就是需要特殊判断一下,不能通过从父亲下来的那条边走上去。(因为强连通分量是有向图,走不上去,所以不存在这个问题)

实现起来也很简单,只用简单修改一下dfs函数:

dfs(u,f)其中u为当前节点,f为走到这个点通过的边的编号。

当在遍历u的所有边试图向下dfs时,只需要加一个if(现在准备选择的边的编号==f)continue;即可。

点双

咕咕咕

Tarjan算法求强连通分量(针对有向图)

stack<int> tp;
void dfs(int u)
{
dfn[u]=low[u]=++cnt1;//初始化每个未访问过的节点
tp.push(u);
for(int i=head[u];i;i=side[i].next)
{
int v=side[i].v;
if(!dfn[v])dfs(v),low[u]=min(low[u],low[v]);
else if(!scc[v])low[u]=min(low[u],dfn[v]);//被访问过却没有SCC编号(在栈里),证明在同一个强连通分量。因为栈维护的是一条有一个节点到它儿子的路径。所以如果栈顶的点u到栈中任意一点v有边。就证明存在u到v的一个环。
}
if(dfn[u]==low[u])//如果这个点是它所在强连通分量中dfn最小的,则有它来承担输出整个SCC的任务
{
int s=tp.top(),id=++cnt2;tp.pop();
scc[s]=id;
while(s!=u)s=tp.top(),tp.pop(),scc[s]=id;
}
}

参考资料

全网最!详!细!Tarjan算法讲解

割点和桥

[算法模版]Tarjan爷爷的几种图论算法的更多相关文章

  1. java开发过程中几种常用算法

    排序算法 排序算法中包括:简单排序.高级排序 简单排序 简单排序常用的有:冒泡排序.选择排序.插入排序 冒泡排序代码如下: private static void bubbleSrot(int[] a ...

  2. Java 的八种排序算法

    Java 的八种排序算法 这个世界,需要遗忘的太多. 背景:工作三年,算法一问三不知. 一.八种排序算法 直接插入排序.希尔排序.简单选择排序.堆排序.冒泡排序.快速排序.归并排序和基数排序. 二.算 ...

  3. 图论算法-Tarjan模板 【缩点;割顶;双连通分量】

    图论算法-Tarjan模板 [缩点:割顶:双连通分量] 为小伙伴们总结的Tarjan三大算法 Tarjan缩点(求强连通分量) int n; int low[100010],dfn[100010]; ...

  4. 图论算法》关于tarjan算法两三事

    关于tarjan,在下觉得这个算法从本质上是一种暴力求强连通分量的方法,但事实上这也是最有效的求强连通分量的方法之一,它对于处理各种强连通分量中奇怪问题,都可以直接转化,所以比较通用和常见. 什么是t ...

  5. tarjan图论算法

    tarjan图论算法 标签: tarjan 图论 模板 洛谷P3387 [模板]缩点 算法:Tarjan有向图强连通分量+缩点+DAGdp 代码: #include <cstdio> #i ...

  6. LCA算法解析-Tarjan&倍增&RMQ

    原文链接http://www.cnblogs.com/zhouzhendong/p/7256007.html UPD(2018-5-13) : 细节修改以及使用了Latex代码,公式更加美观.改的过程 ...

  7. Kosaraju算法、Tarjan算法分析及证明--强连通分量的线性算法

    一.背景介绍 强连通分量是有向图中的一个子图,在该子图中,所有的节点都可以沿着某条路径访问其他节点.强连通性是一种非常重要的等价抽象,因为它满足 自反性:顶点V和它本身是强连通的 对称性:如果顶点V和 ...

  8. 对比几种在ROS中常用的几种SLAM算法

    在此因为要总结写一个文档,所以查阅资料,将总结的内容记录下来,欢迎大家指正! 文章将介绍使用的基于机器人操作系统(ROS)框架工作的SLAM算法. 在ROS中提供的五种基于2D激光的SLAM算法分别是 ...

  9. JVM的分区+查看GC对象是否存活+3种GC算法+7种垃圾收集器+如何减少GC次数

    一.JVM的分区:   1.程序计数器(私有) 程序计数器是一块较小的内存分区,你可以把它看做当前线程所执行的字节码的指示器. 在虚拟机的概念模型里,字节码解释器工作时,就是通过改变计数器的值来选择下 ...

随机推荐

  1. Ubuntu Idea 快捷键 Ctrl+Alt+S 无法使用解决

    Idea 里习惯了用 Ctrl+Alt+S 打开设置界面,在 Ubuntu 下会因为快捷键冲突无法使用 系统快捷键 打开系统设置中的快捷键设置,按 Backspace 键禁用 Fcitx 如果你的输入 ...

  2. JMeter 使用 http长连接 |史上最全

    疯狂创客圈 Java 高并发[ 亿级流量聊天室实战]实战系列 [博客园总入口 ] 疯狂创客圈(笔者尼恩创建的高并发研习社群)Springcloud 高并发系列文章,将为大家介绍三个版本的 高并发秒杀: ...

  3. spring cloud 2.x版本 Gateway动态路由教程

    摘要 本文采用的Spring cloud为2.1.8RELEASE,version=Greenwich.SR3 本文基于前面的几篇Spring cloud Gateway文章的实现. 参考 Gatew ...

  4. 【Linux命令】文件目录管理命令7个(touch、mkdir、cp、mv、rm、dd、file)

    目录 touch创建空白文档或设置文件时间 mkdir创建空白目录 cp复制文件或目录 mv剪切文件或重命名文件 rm删除文件或目录 dd按照指定大小和个数的数据库来复制文件或转换文件 file查看文 ...

  5. Selenium+java - 关于富文本编辑器的处理

    什么是富文本编辑器? 富文本编辑器,Rich Text Editor, 简称 RTE, 是一种可内嵌于浏览器,所见即所得的文本编辑器.具体长啥样,如下图: 通过自动化操作富文本编辑器 模拟场景:在富文 ...

  6. Protractor - 怎样运行

    前一篇设置好了Protractor基本运行环境,那怎样运行Protractor呢? 要运行我们的测试脚本,至少需要配置好两个文件: ---Package.json ---conf.js Package ...

  7. gcc 编译安装

    wget https://kojipkgs.fedoraproject.org//packages/gcc/7.1.1/1.module_1a179a7b/src/gcc-7.1.1-1.module ...

  8. 在.net core程序中使用EntityFrameok(非EF Core)

    最近用NoSQL较多写,用传统的EF到不多,但在一些.net core小程序中也小试牛刀过,不过当时用的是微软为.net core量身定制的Entity Framework Core,只是一些比较常规 ...

  9. frigate_TUNNEL

    #coding=utf-8 Result=open('result.txt',"w") FileTunnel = open('tunnel.txt').readlines() Ne ...

  10. sed文本处理

    1.基本概述 sed是一个流编辑器, 非交互式的编辑器,它一次处理一行内容. 处理时,把当前处理的行存储在临时缓冲区中,称* 为"模式空间"(pattern space) 接着用 ...