知识点-Tarjan

强连通分量:在一个图的子图中,任意两个点相互可达,也就是存在互通的路径,那么这个子图就是强连通分量(或者称为强连通分支)。如果一个有向图的任意两个点相互可达,那么这个图就称为强连通图

当我们实现基于dfs的Tarjan算法时,我们用D[i]记录节点i被访问的时间(别的博客用dfn[i]),用F[i]记录节点i或i的子树最小可以返回到的节点j的D[j](别的博客用low[i])。

让我们模拟一下dfs的过程,每个节点上左边的是D的值,右边的是F的值。

首先从节点1开始往下深搜,每次搜索都更新D值和F值,此时F值是节点本身。

当搜索到节点4时,已经无法继续搜索。而此时D值=F值,所以我们可以判定一个强连通分量(在代码中,需要借助“栈”来寻找强连通分量中所有节点),这个强连通量就是节点4。

从图我们也可以看出,节点4无法到达其它任意节点。

从节点4返回至3,此时节点3也无路可走了。而D值=F值,那么又可以判定一个强连通分量节点3。

从3返回至2,节点2还有一条路可走!深搜节点5,修改其D值和F值。

节点5有路通向节点1,发现节点1的值已经被修改过了的!节点5的F值要修改成1,也就是它到达D值最远的节点是1。修改完后无需继续深搜节点1否则会造成死循环!而D值不等于F值,所以返回。

修改节点2的F值后,再返回。

从节点2返回至1,还有一条路通往节点6。深搜节点6并修改6的D值和F值。

节点6有路通向节点5,发现节点5的值已经被修改过!于是修改节点6的F值后返回至节点1。

当返回到节点1时,已经无路可走,D值等于F值,所以又一个强连通分量出来了……

以上的过程都可以用代码来实现。要注意F值的修改:

如果节点i的子节点j的值没有被修改,那么F[i]=min(F[i],F[j]);

否则F[i]=min(F[i],D[j]);

例题

题目描述

求强连通分量……

输入

第一行两个整数,n,m,代表点数及边数。

第2行至m+1行,每行两个整数,a,b,代表有向边的起点和终点。

输出

输出所有强连通分量

样例输入

6 8
1 2
2 3
3 6
5 6
2 5
5 1
1 4
4 5

样例输出

6
3
2 5 4 1

代码

 #include<cstdio>
#include<algorithm>
using namespace std;
struct point{
int nxt,to;
}edge[];
int d[],f[],stack[],head[]={};
bool vis[];
int tol,cnt,n,m,a,b,index;
void add(int u,int v)
{
edge[++cnt].to=v;
edge[cnt].nxt=head[u];
head[u]=cnt;
}
int tarjan(int node)
{
tol++;
d[node]=f[node]=tol;
stack[++index]=node;
vis[node]=;
for(int i=head[node];i!=;i=edge[i].nxt)
{
if(!d[edge[i].to])
{
tarjan(edge[i].to);
f[node]=min(f[edge[i].to],f[node]);
}
else if(vis[edge[i].to])
f[node]=min(d[edge[i].to],f[node]);
}
if(d[node]==f[node])
{
do
{
printf("%d",stack[index]);
if(node!=stack[index])printf(" ");
vis[stack[index]]=;
index--;
}while(node!=stack[index+]);
printf("\n");
}
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=;i<=m;i++)
{
scanf("%d%d",&a,&b);
add(a,b);
}
for(int i=;i<=n;i++)
if(!d[i])tarjan(i);
return ;
}

Tarjan-求强连通分量的更多相关文章

  1. UESTC 901 方老师抢银行 --Tarjan求强连通分量

    思路:如果出现了一个强连通分量,那么走到这个点时一定会在强连通分量里的点全部走一遍,这样才能更大.所以我们首先用Tarjan跑一遍求出所有强连通分量,然后将强连通分量缩成点(用到栈)然后就变成了一个D ...

  2. tarjan求强连通分量+缩点+割点以及一些证明

    “tarjan陪伴强联通分量 生成树完成后思路才闪光 欧拉跑过的七桥古塘 让你 心驰神往”----<膜你抄>   自从听完这首歌,我就对tarjan开始心驰神往了,不过由于之前水平不足,一 ...

  3. Tarjan求强连通分量,缩点,割点

    Tarjan算法是由美国著名计算机专家发明的,其主要特点就是可以求强连通分量和缩点·割点. 而强联通分量便是在一个图中如果有一个子图,且这个子图中所有的点都可以相互到达,这个子图便是一个强连通分量,并 ...

  4. tarjan求强连通分量+缩点+割点/割桥(点双/边双)以及一些证明

    “tarjan陪伴强联通分量 生成树完成后思路才闪光 欧拉跑过的七桥古塘 让你 心驰神往”----<膜你抄>   自从听完这首歌,我就对tarjan开始心驰神往了,不过由于之前水平不足,一 ...

  5. HDU 1827 Summer Holiday(tarjan求强连通分量+缩点构成新图+统计入度+一点贪心思)经典缩点入门题

    Summer Holiday Time Limit: 10000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)T ...

  6. CCF 高速公路 tarjan求强连通分量

    问题描述 某国有n个城市,为了使得城市间的交通更便利,该国国王打算在城市之间修一些高速公路,由于经费限制,国王打算第一阶段先在部分城市之间修一些单向的高速公路. 现在,大臣们帮国王拟了一个修高速公路的 ...

  7. UVALive 4262——Trip Planning——————【Tarjan 求强连通分量个数】

    Road Networks Time Limit:3000MS     Memory Limit:0KB     64bit IO Format:%lld & %llu Submit Stat ...

  8. tarjan求强连通分量(模板)

    https://www.luogu.org/problem/P2341 #include<cstdio> #include<cstring> #include<algor ...

  9. Tarjan求强连通分量、求桥和割点模板

    Tarjan 求强连通分量模板.参考博客 #include<stdio.h> #include<stack> #include<algorithm> using n ...

  10. poj 2186 tarjan求强连通分量

    蕾姐讲过的例题..玩了两天后才想起来做 貌似省赛之后确实变得好懒了...再努力两天就可以去北京玩了! 顺便借这个题记录一下求强连通分量的算法 1 只需要一次dfs 依靠stack来实现的tarjan算 ...

随机推荐

  1. java基础:数组的拼接

  2. wemall app商城源码机器人检测

    wemall-mobile是基于WeMall的Android app商城,只需要在原商城目录下上传接口文件即可完成服务端的配置,客户端可定制修改.本文分享wemall app商城源码Android之 ...

  3. 添加swagger api文档到node服务

    swagger,一款api测试工具,详细介绍参考官网:http://swagger.io/ ,这里主要记录下怎么将swagger api应用到我们的node服务中: 1.任意新建node api项目, ...

  4. 解决 Linux error while loading shared libraries: cannot open shared object file: No such file or directory

    安装最新版本Emqtt,参照官方文档安装后,执行报错: Linux error while loading shared libraries libsctp.so.1: cannot open sha ...

  5. (9)集合之Set,HashSet,TreeSet

    TreeSet子类 注意事项: 1.向TreeSet添加元素的时候,如果元素本身具备了自然顺序的特性,那么就按照元素自然顺序的特性进行排序存储 2.往TreeSet添加元素的时候,如果元素本身不具备自 ...

  6. C#传递委托给C或C++库报错__对XXX类型的已垃圾回收委托进行了回调

    出现的原因: 因为你传给C或C++的委托是局部的.可能传过去之后就被垃圾回收了,再次调用就会异常. 想办法做成全局的就好 public void Play(string url) { _bassStr ...

  7. Hive 伪分布式下安装

    本安装过程只作为个人笔记用,非标准教程,请酌情COPY.:-D Hive下载 下载之前,需先查看兼容的Hadoop版本,并安装hadoop,参考 http://www.cnblogs.com/yong ...

  8. centos安装python2.7并安装easy_install,pip,ipython

    1.安装python 下载python2.7.10 # wget https://www.python.org/ftp/python/2.7.10/Python-2.7.10.tgz # tar -z ...

  9. unity插件开发——MenuItem

    有unity中的菜单栏是我们经常使用到的地方,如下图: MenuItem的作用就是增加一个自己的菜单 使用方法: 在工程中Assets目录下任意一个Editor目录(以后简称Editor目录,如果不存 ...

  10. OC--Runtime知识点整理

    1.Runtime简介 因为Objc是一门动态语言,所以它总是想办法把一些决定工作从编译连接推迟到运行时.也就是说只有编译器是不够的,还需要一个运行时系统 (runtime system) 来执行编译 ...