tarjan算法比较详细的讲解&&tarjan常见疑难解答&&洛谷P2002 消息扩散题解
因为有大佬写的比我更长更具体,所以我也就写写总结一下了
引入:
众所周知,很多图中有个东西名叫环。
对于这个东西很多算法都很头疼。(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 消息扩散题解的更多相关文章
- MySQL 主从复制 详细实例讲解 与 常见错误解决方法
一.主机ip 192.168.0.128 ,从机ip:192.168.0.130 分别测试是否能ping通对方,如果不能,请关闭防火墙或开放对应端口 二.主服务器配置 1.备份主服务器的数据 mysq ...
- 【算法•日更•第六期】头脑风暴:洛谷P1528 切蛋糕题解
▎(一个没有用处的)前言 为什么这次题解特意写明题号呢?因为我发现了这样的事情: 所以不要混了,想看P1714题解的同志们可以圆润的滚开了. 好了,不说没用的了,切入正题: ▎题目 题目及测评链接:戳 ...
- tarjan 算法讲解(转)
转自:https://www.byvoid.com/blog/scc-tarjan/ 網誌 列表 標籤 項目 關於 聯繫 四月142009 作者:byvoid 閱讀: 158882 計算機科學 圖論 ...
- 有向图强连通分支的Tarjan算法讲解 + HDU 1269 连通图 Tarjan 结题报告
题目很简单就拿着这道题简单说说 有向图强连通分支的Tarjan算法 有向图强连通分支的Tarjan算法伪代码如下:void Tarjan(u) {dfn[u]=low[u]=++index//进行DF ...
- 有向强连通分支Tarjan算法
本文转载自:http://blog.csdn.net/xinghongduo/article/details/6195337 说到以Tarjan命名的算法,我们经常提到的有3个,其中就包括本文所介绍的 ...
- tarjan算法大意
Tarjan算法 (以发现者Robert Tarjan命名)是一个在图中寻找强连通分量的算法.算法的基本思想为:任选一结点开始进行深度优先搜索dfs(若深度优先搜索结束后仍有未访问的结点,则再从中任选 ...
- Tarjan 算法求强联通分量
转载自:http://blog.csdn.net/xinghongduo/article/details/6195337 还是没懂Tarjan算法的原理.但是感觉.讲的很有道理. 说到以Tarjan命 ...
- 割点(Tarjan算法)【转载】
本文转自:www.cnblogs.com/collectionne/p/6847240.html 供大家学习 前言:之前翻译过一篇英文的关于割点的文章(英文原文.翻译),但是自己还有一些不明白的地方, ...
- Tarjan算法 详解+心得
Tarjan算法是由Robert Tarjan(罗伯特·塔扬,不知有几位大神读对过这个名字) 发明的求有向图中强连通分量的算法. 预备知识:有向图,强连通. 有向图:由有向边的构成的图.需要注意的是这 ...
随机推荐
- 【JVM学习笔记】双亲委托机制存在的意义
1.可以确保Java核心库的类型安全:所有的Java应用都至少会引用java.lang.Object类,也就是说在运行期,java.lang.Object这个类会被加载到Java虚拟机:如果用户自定义 ...
- Using NodeLists
Understanding a NodeList object and its relatives, NamedNodeMap and HTMLCollection, is critical to a ...
- 【HANA系列】SAP HANA查看某一用户最后登录时间及无效连接次数
公众号:SAP Technical 本文作者:matinal 原文出处:http://www.cnblogs.com/SAPmatinal/ 原文链接:[HANA系列]SAP HANA查看某一用户最后 ...
- 模块的概念、模块的导入方式【IMPORT 模块名、FROM 模块 IMOPRT 功能】、模块的搜索路径、链式导入&循环导入
今日内容 1. 模块:模块的概念 2.导入的方式:import from import 3. 环境变量:sys.path 4. 导入模块的顺序 5. 循环导入:模块间互相导入 模块 常见的四种模块: ...
- 【DSP开发】【计算机视觉】TI 视觉软件开发套件ADAS
关键字:TI 视觉软件开发套件 ADAS 日前,德州仪器 (TI) 宣布推出其视觉软件开发套件(SDK),从而为开发人员提供了一款灵活的框架.一组丰富齐全的硬件设备驱动程序和一套适用的开发工具,可 ...
- 解决应用程序无法正常启动0xc0150002问题(转)
简述:使用VS2008写了一个MFC程序,结果传到别人的机子上(WIN7)出现应用程序正常初始化(0xc0150002)失败的问题.为什么我的机子上可以,而别人的机子上运行不了呢?下面是我找到的一个解 ...
- 西安邀请赛-E(树链剖分+线段树)
题目链接:https://nanti.jisuanke.com/t/39272 题意:给一棵树,n个结点,树根为1,n-1条边,每个结点有一个权值.进行3种操作: 1 s t:把1和s之间的最短路径上 ...
- Eclipse中导入Maven Web项目并配置其在Tomcat中运行
今天因为实习的关系需要讲公司已经开发的项目导入进Eclipse,而公司的项目是用Maven来构建的所以,需要将Maven项目导入进Eclipse下. 自己因为没有什么经验所以搞了得两个多小时,在这里和 ...
- 坦克大战--Java类型 ---- (2)按键设置和用户名的输入
一.实现思路(emmmm,这个地方我很大程度参照了别人的写法) 由于键盘按键众多,因此使用选择框JComboBox 进行按键选择,点击一个JButton 按钮后,读取所有选择框中的内容,然后存到一 ...
- Adam作者大革新, 联合Hinton等人推出全新优化方法Lookahead
Adam作者大革新, 联合Hinton等人推出全新优化方法Lookahead 参与:思源.路.泽南 快来试试 Lookahead 最优化方法啊,调参少.收敛好.速度还快,大牛用了都说好. 最优化方 ...