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(罗伯特·塔扬,不知有几位大神读对过这个名字) 发明的求有向图中强连通分量的算法. 预备知识:有向图,强连通. 有向图:由有向边的构成的图.需要注意的是这 ...
随机推荐
- linux常用、常见错误
1.md5加密使用 oppnssl md5 加密字符串的方法 [root@lab3 ~]# openssl //在终端中输入openssl后回车. OpenSSL> md5 //输入md5后回车 ...
- Qt输出中文乱码的问题
/* 我遇到的情况: 文件编码为UTF-8.程序输输出中文位乱码. 解决方案: 1. 工具->选项->环境->语言: Chinese 2. 选项->文本编辑器->行为-& ...
- Unity中的动画系统和Timeline(2) 按钮动画和2D精灵动画
按钮动画 1 创建按钮后,按钮的Button组件中,Transition我们平时用的时Tint,这次选择Animation 选择Auto Generate Animation,创建一个按钮动画 2 后 ...
- js动态修改浏览器title
script标签依据浏览框的失焦获焦进行函数操作(操作简单放到HTML文件下的head标签下就可以) <script> window.onfocus = function () { doc ...
- @Value注解
1.注入 基本字符 public class Student { @Value("qq") private String name; @Value("12") ...
- 人工智能01 刺激响应agent
刺激响应agent 不具有内部状态而仅对其所处环境的即刻刺激有所反应的机器称为刺激响应(SR)agent 感知和动作 一机器人可以感知出周围8个单元是否空缺.这些传感器输入用二进制变量s1,s2 ,s ...
- JAVA -数据类型与表达式---基本数据类型
基本数据类型 Java有8种基本数据类型(primitive data type):4种整型.2种浮点型.字符型和布尔型.除此之外的任何类型都用对象表示.本节将详细讨论上述8种基本数据类型. 整型与浮 ...
- 编译安装php7.3
./configure --prefix=/usr/local/php7.3.9 --with-gd --enable-mysqlnd --with-mysqli=mysqlnd --with-pdo ...
- Python 入门 之 异常处理
Python 入门 之 异常处理 1.异常处理 (1)程序中的错误分为两种 <1> 语法错误 (这种错误,根本过不了Python解释器的语法检测,必须在程序执行前就改正) # 语法错误示范 ...
- java基础:强引用、弱引用、软引用和虚引用 (转)
出处文章: Java基础篇 - 强引用.弱引用.软引用和虚引用 谈谈Java对象的强引用,软引用,弱引用,虚引用分别是什么 整体结构 java提供了4中引用类型,在垃圾回收的时候,都有自己的各自特点. ...