因为有大佬写的比我更长更具体,所以我也就写写总结一下了

引入:

众所周知,很多图中有个东西名叫环。

对于这个东西很多算法都很头疼。(suchas 迪杰斯特拉)

更深层:环属于强联通分量(strongly connected components)

定义:如果有向图G的每两个顶点都强连通,称G是一个强连通图。有向图的极大强连通子图,称为强连通分量。

例如:(画画技艺高超到自闭)

红圈内部即为强联通分量。

对于这个东西,其他算法更难搞;

那么我们为了把这个东西合为一个,或者找出属于一个强联通分量的所有元素,或者其他某些操作,一个叫tarjan的人研发了一个tarjan算法。

为他正名:tarjan,自己听怎么读。

对于tarjan算法求强联通分量,步骤:

1.定义两个数组dfn和low,

dfn[x]表示x节点是第几个被遍历到的。

low[x]表示x以及它的所有子树的出边的dfn的最小值。

2,我们用一个栈来存储遍历到的点,把当前搜到的点的入栈标记in记为true。

3,对于每一个当前节点的子节点,如果之前没有遍历到,就往下遍历,并在遍历后取low的min值:low[x]=min(low[x],low[v])。如果遍历到之前已经遍历到的,也就是in[v]=1,那么我们就不再往下继续搜了,直接传递dfn的值:low[x]=min(low[x],dfn[v]);

4,如果当前节点的dfn的值等于low的值,由于low表示当前节点以及其所有子树的出边的最小dfn的值,那么我们就可以认为low被传递了一圈又回到了当前dfn,也就代表找到了一个强联通分量,那么我们就把从当前的栈顶一直到当前元素的栈中元素全部弹出并作为强联通分量的一部分。

算法主体框架完毕,本质为深搜。

特别的,一个节点也算一个强联通分量,因为它能自己到自己。。。

上代码:

void tarjan(int u){
dfn[u]=++ind;
low[u]=dfn[u];//dfn和low初始化,low的初始值为当前dfn,low可以被其子节点更新。
s[top++]=u;//s代表栈,用来存强联通分量。
in[u]=;//in入栈标记,为bool数组
for(int i=head[u];i;i=e[i].next){//链式前向星存图
int v=e[i].to;//出边
if(dfn[v]==){//如果没有到过
tarjan(v);//继续往下搜
low[u]=min(low[u],low[v]);//更新low的值
}else{//bian li dao le, v bu zai zi shu li mian
if(in[v]){//zai zhan li mian
low[u]=min(low[u],dfn[v]);//更新,但是为什么为dfn而不是low,下文详解
}
}
}
if(dfn[u]==low[u]){//为强联通分量的根节点
cnt_scc++;//用来标记为当前强联通分量的编号
while(s[top]!=u){//不断弹出一直弹到当前节点
top--;
in[s[top]]=;
scc[s[top]]=cnt_scc;//tarjan主要作用,标记当前点属于第几个强联通分量(也可以作为缩点)
}
}
}

那么,洛谷p2002这道题也就很好做了。

思路:

显然,对于每一个强联通分量内部来说,只要有一个点能够被传送到消息,那么其他点都能被传送到消息。

那么我们将所有强联通分量缩为一个点,然后统计所有强联通分量的入度,如果为0那么答案就需要加一来保证当前强联通分量以及它的子节点们能够被传递到信息。

code:

#include<cstdio>
#include<iostream>
using namespace std; int n
int m;
int fro;
int t;
int sccnum=;
int ru[];//表示 i这个强联通分量有无入度(bool也可以)
int num=;
int ans=;
int dfn[];
int low[];
int head[];
int stack[];
int top;
int cnt=;
int belong[];
bool in[]; struct edge{
int to,next;
}edg[]; inline int read()//自研快读
{
int ans=;
char ch=getchar(),last=' ';
while(ch<''||ch>'')last=ch,ch=getchar();
while(ch>=''&&ch<='')ans=(ans<<)+(ans<<)+ch-'',ch=getchar();
return last=='-'?-ans:ans;
} inline void add(int from,int to)//链式存图
{
num++;
edg[num].to=to;
edg[num].next=head[from];
head[from]=num;
} void tarjan(int x)//主体部分
{
stack[top++]=x;
dfn[x]=low[x]=++cnt;
in[x]=;
for(int i=head[x];i;i=edg[i].next)
{
int v=edg[i].to;
if(!dfn[v])
{
tarjan(v);
low[x]=min(low[x],low[v]);
}
else if(in[v])
{
low[x]=min(low[x],dfn[v]);
}
}
if(dfn[x]==low[x])
{
sccnum++;
while(stack[top]!=x)
{
top--;
in[stack[top]]=;
belong[stack[top]]=sccnum;//标记为一个强联通分量
}
}
} int main(){
n=read(),m=read();
for(int i=;i<=m;i++)
{
fro=read(),t=read();
if(fro!=t)add(fro,t);
}
for(int i=;i<=n;i++)//对每一个点跑一个tarjan
{
if(!dfn[i])tarjan(i);
}
for(int i=;i<=n;i++)
{
for(int j=head[i];j;j=edg[j].next)//遍历每一个点统计强联通分量入度
{
int v=edg[j].to;
if(belong[i]!=belong[v])ru[belong[v]]=;
}
}
for(int i=;i<=sccnum;i++)//如果入度为0,答案加一
{
if(!ru[i])ans++;
}
printf("%d",ans);//完结
return ;
}

文章主体部分完结。

关于dfn和low的争议:在low取min的时候,为什么不用low[v]而用dfn[v]?

这个也是争论很久的问题,其中著名北大金牌dms就犯错犯过一段时间并且毫无察觉直到他做一个题WA掉。。。

我的理解是这样的。

我们注意到low的定义:low[x]是x的子树里最小的dfn。

如果已经访问过,那么它一定是当前节点的祖先。

如果用low的话那么它也就不再符合定义。

用dfn的话不影响low的传递。因为low是传递一圈最终传递回强联通分量的根节点,用dfn并不影响它祖先的low值传递向其他边,只是作为一个low传递路径的完结标记。(十分感性理解)。

况且在算割点的时候用low的话就把整个强联通分量同化了(low是层层传递的,感性理解),那么求割点的正确性就发生了变化。所以low是不正确的。

当然,因为图论题数据比较难造,有些题目数据不强导致你用low就可以过。况且用low的话对缩点影响不如割点大。

感性理解完,如果你感性不强,我也没办法QWQ。

完结辣。

对各位有帮助理解的话顶一个吧。

tarjan算法比较详细的讲解&&tarjan常见疑难解答&&洛谷P2002 消息扩散题解的更多相关文章

  1. MySQL 主从复制 详细实例讲解 与 常见错误解决方法

    一.主机ip 192.168.0.128 ,从机ip:192.168.0.130 分别测试是否能ping通对方,如果不能,请关闭防火墙或开放对应端口 二.主服务器配置 1.备份主服务器的数据 mysq ...

  2. 【算法•日更•第六期】头脑风暴:洛谷P1528 切蛋糕题解

    ▎(一个没有用处的)前言 为什么这次题解特意写明题号呢?因为我发现了这样的事情: 所以不要混了,想看P1714题解的同志们可以圆润的滚开了. 好了,不说没用的了,切入正题: ▎题目 题目及测评链接:戳 ...

  3. tarjan 算法讲解(转)

     转自:https://www.byvoid.com/blog/scc-tarjan/ 網誌 列表 標籤 項目 關於 聯繫 四月142009 作者:byvoid 閱讀: 158882 計算機科學 圖論 ...

  4. 有向图强连通分支的Tarjan算法讲解 + HDU 1269 连通图 Tarjan 结题报告

    题目很简单就拿着这道题简单说说 有向图强连通分支的Tarjan算法 有向图强连通分支的Tarjan算法伪代码如下:void Tarjan(u) {dfn[u]=low[u]=++index//进行DF ...

  5. 有向强连通分支Tarjan算法

    本文转载自:http://blog.csdn.net/xinghongduo/article/details/6195337 说到以Tarjan命名的算法,我们经常提到的有3个,其中就包括本文所介绍的 ...

  6. tarjan算法大意

    Tarjan算法 (以发现者Robert Tarjan命名)是一个在图中寻找强连通分量的算法.算法的基本思想为:任选一结点开始进行深度优先搜索dfs(若深度优先搜索结束后仍有未访问的结点,则再从中任选 ...

  7. Tarjan 算法求强联通分量

    转载自:http://blog.csdn.net/xinghongduo/article/details/6195337 还是没懂Tarjan算法的原理.但是感觉.讲的很有道理. 说到以Tarjan命 ...

  8. 割点(Tarjan算法)【转载】

    本文转自:www.cnblogs.com/collectionne/p/6847240.html 供大家学习 前言:之前翻译过一篇英文的关于割点的文章(英文原文.翻译),但是自己还有一些不明白的地方, ...

  9. Tarjan算法 详解+心得

    Tarjan算法是由Robert Tarjan(罗伯特·塔扬,不知有几位大神读对过这个名字) 发明的求有向图中强连通分量的算法. 预备知识:有向图,强连通. 有向图:由有向边的构成的图.需要注意的是这 ...

随机推荐

  1. JS事件中级 --- 拖拽

    http://bbs.zhinengshe.com/thread-1200-1-1.html 要求:实现div块的拖拽 原理:拖拽过程中鼠标点和div块的相对位置保持不变. 需要理解三点: 1. 为什 ...

  2. oslo_db.sqlalchemy.engines连库

    _ mysql -uroot -pc1234 oslo_db.sqlalchemy.engines root@devstack2019:/etc/keystone# more keystone.con ...

  3. 【学习笔记】python3中yaml文件使用

    1.yaml -> 字典:用yaml.load()或yaml.safe_load(YAML字符串或文件句柄),如yaml中有中文,可以使用.encode('utf-8')或打开文件时指定enco ...

  4. 《Linux命令行大全》 笔记记录

    1.Shell是什么 2.(文件目录)导航 3.Linux系统 4.操作文件和目录 5.命令的使用 6.重定向 7.透过shell看世界 扩展 引用 8.高级键盘技巧 9.权限 10.进程 11.环境 ...

  5. CentOS下Subversion(SVN)的快速安装与配置

    如果你是一个软件开发者,你一定对Subversion不会感到陌生.Subversion是一个自由开源的版本控制系统.在Subversion管理下,文件和目录可以超越时空.Subversion将文件存放 ...

  6. 10分钟学会web通讯的四种方式,短轮询、长轮询(comet)、长连接(SSE)、WebSocket

    一般看到标题我们一般会产生下面几个问题??? 什么是短轮询? 什么是长轮询? 长连接又是什么? wensocket怎么实现呢? 他们都能实现web通讯,区别在哪呢,哪个好用呢? 接下来我们就一个个来了 ...

  7. Asteroid Collision

    We are given an array asteroids of integers representing asteroids in a row. For each asteroid, the ...

  8. [转帖]【JDK和Open JDK】平常使用的JDK和Open JDK有什么区别

    https://www.cnblogs.com/sxdcgaq8080/p/7487369.html 其实不同的 openjdk的版本也不一样. atlassian说AdoptOpenJDK我们测试充 ...

  9. [转帖]Java 8新特性探究(八)精简的JRE详解

    Java 8新特性探究(八)精简的JRE详解 https://my.oschina.net/benhaile/blog/211804 精简版的api   撸了今年阿里.网易和美团的面试,我有一个重要发 ...

  10. Java代码执行顺序及多态体现

    /** * Description: * 基类的引用变量可以只想基类的实例对象也可指向其子类的事来对象 * 接口的引用变量也可以指向实现类的实例对象 * 程序调用的方法在运行期才动态绑定 * 绑定指将 ...