变量解释:

low 指当前节点在同一强连通分量(或环)能回溯到的dfn最小的节点

dfn 指当前节点是第几个被搜到的节点(时间戳)

sta 栈

vis 是否在栈中

ans 指强连通分量的数量

top 栈顶

1.求强连通分量

定义:如果两个顶点可以相互通达,则称两个顶点强连通(strongly connected)。如果有向图G的每两个顶点都强连通,称G是一个强连通图。有向图的极大强连通子图,称为强连通分量(strongly connected components)。

算法:在有向图中从一点(u)开始dfs,记录dfn,搜到一个已在栈中的点(v)时用dfn[v] (low[v]也行,但只有求强连通分量时可以别的只能用dfn[v]) 尝试更新low[u],并在回溯时更新沿路的点的low值,走到low值与dfn相同的点时记录这个强连通分量即可。

也就是说:在同一个强连通分量中所有点low值相同,也就是有一个代表点(代表点即所有点的low值即强连通分量中dfn值最小的点)

时间复杂度为O(E+V)

code

void tarjan(int u){
dfn[u]=low[u]=++cnt;//初始化一点的dfn和low
sta[++top]=u,vis[u]=true;//入栈
for(int i=head[u];i;i=edge[i].next){//邻接表
int v=edge[i].to;
if(!dfn[v]){//如果没走过
tarjan(v);
low[u]=min(low[u],low[v]);//回溯过程时low值传递
}
else if(vis[v]) low[u]=min(low[u],dfn[v]); //low[v]也行 用代表点更新
}
if(dfn[u]==low[u]) {//如果是代表点 记录并出栈
ans++;//记录强连通分量个数
while(sta[top]!=u){
vis[sta[top]]=false;
top--;
}
vis[sta[top]]=false;
top--;
}
return ;
}

2.求无向图的割点与割边

割点:在无向图中,如果将一个点以及所有连接该点的边都去掉,图就不再连通,那么这个点就叫做这个图的一个割点。

割边:在无向图中,如果将一条边去掉,图就不再连通则称这条边为图的一个割边。

求割点:如果一个点(u)所连接的几个节点(v)的low值大或等于此节点(u)的dfn值时说明之后的节点(v)无法连接到比此点(u)更早的点上,则说明这个节点(u)是一个割点。PS:根节点需特判,当根节点在dfs树有两个或更多个子树时则说明根节点是割点

求割边:与割点类似,如果一个点(u)的dfn值大于(不能等于,否则不一定)和它连接的一个节点(v)的low值,则说明这条边(uv)为图的一个割边

变量解释:

sum 指总共有几个割点(边)

割点code

void cutpoint(int u){
int fl=0;//为特判准备
dfn[u]=low[u]=++cnt;//初始化
for(int i=head[u];i;i=edge[i].next){//用邻接表,下同
int v=edge[i].to;
if(!dfn[v]){
cutpoint(v);
low[u]=min(low[u],low[v]);
if(u!=root&&low[v]>=dfn[u]&&!cpoint[u]) sum++,cpoint[u]=1;//不是根节点&&v的low值>=u的dfn值&&此点没有算过
if(u==root) fl++;//此时特判++
}
low[u]=min(low[u],dfn[v]);
}
if(fl>=2&&!cpoint[u]) sum++,cpoint[u]=1;//根节点若有两棵子树则是割点
}

割边code

void cutedge(int u,int f){
dfn[u]=low[u]=++cnt;
for(int i=head[u];i;i=edge[i].next){
int v=edge[i].to;
if(!dfn[v]){
cutedge(v,u);
low[u]=min(low[u],low[v]);
if(dfn[u]<low[v]) cedge[++sum]=i;//记录边的序号
}
else if(v!=f) low[u]=min(low[u],dfn[v]); //只有当v不是u的上一个节点时可行
}
}

完整模板code:

ps:这里就不打注释了,核心就在上面的部分里

#include<cstdio>
#include<iostream>
#include<cstring>
using namespace std; const int MAX=1000010;
int n,m,cnt,sum,root;
int head[MAX],low[MAX],dfn[MAX],cpoint[MAX],cedge[MAX]; struct edg{
int to,next,from;
}edge[MAX]; void add(int x,int y){
edge[++cnt].next=head[x];
edge[cnt].from=x,edge[cnt].to=y;
head[x]=cnt;
} void cutpoint(int u){
int fl=0;
dfn[u]=low[u]=++cnt;
for(int i=head[u];i;i=edge[i].next){
int v=edge[i].to;
if(!dfn[v]){
cutpoint(v);
low[u]=min(low[u],low[v]);
if(u!=root&&low[v]>=dfn[u]&&!cpoint[u]) sum++,cpoint[u]=1;
if(u==root) fl++;
}
low[u]=min(low[u],dfn[v]);
}
if(fl>=2&&!cpoint[u]) sum++,cpoint[u]=1;
} void cutedge(int u,int f){
dfn[u]=low[u]=++cnt;
for(int i=head[u];i;i=edge[i].next){
int v=edge[i].to;
if(!dfn[v]){
cutedge(v,u);
low[u]=min(low[u],low[v]);
if(dfn[u]<low[v]) cedge[++sum]=i;
}
else if(v!=f) low[u]=min(low[u],dfn[v]);
}
} void mset(){
memset(dfn,0,sizeof dfn);
memset(low,0,sizeof low);
cnt=sum=0;
} void find_cutpoint(){
for(int i=1;i<=n;i++) if(!dfn[i]) {
root=i;
cutpoint(i);
}
printf("%d\n",sum);
for(int i=1;i<=n;i++) if(cpoint[i]) printf("%d ",i);
} void find_cutedge(){
for(int i=1;i<=n;i++) if(!dfn[i]) cutedge(i,0);
printf("%d\n",sum);
for(int i=1;i<=sum;i++) printf("%d %d\n",edge[cedge[i]].from,edge[cedge[i]].to);
} int main(){
// freopen("testdata.txt","r",stdin);
scanf("%d %d",&n,&m);
for(int i=1;i<=m;i++){
int a,b;
scanf("%d %d",&a,&b);
add(a,b);
add(b,a);
}
find_cutedge();
mset();
find_cutpoint();
return 0;
}

Tarjan算法 (强联通分量 割点 割边)的更多相关文章

  1. tarjan模板 强联通分量+割点+割边

    // https://www.cnblogs.com/stxy-ferryman/p/7779347.html ; struct EDGE { int to, nt; }e[N*N]; int hea ...

  2. Tarjan算法---强联通分量

    1.基础知识 在有向图G,如果两个顶点间至少存在一条路径,称两个顶点强连通(strongly connected).如果有向图G的每两个顶点都强连通,称G是一个强连通图.非强连通图有向图的极大强连通子 ...

  3. tarjan求强联通分量

    tarjan求强联通分量 变量含义说明: pre[i]:i点的被访问的时钟编号,被分配后保持不变 low[i]:i点能访问的最先的点的时钟编号,随子节点改变 scc_no[i]:i点所在的强联通分量的 ...

  4. Kosaraju算法---强联通分量

    1.基础知识 所需结构:原图.反向图(若在原图中存在vi到vj有向边,在反向图中就变为vj到vi的有向边).标记数组(标记是否遍历过).一个栈(或记录顶点离开时间的数组).      算法描叙: :对 ...

  5. Tarjan的强联通分量

    求强联通分量有很多种. <C++信息学奥赛一本通>  中讲过一个dfs求强联通分量的算法Kosdaraju,为了骗字数我就待会简单的说说.然而我们这篇文章的主体是Tarjan,所以我肯定说 ...

  6. Tarjan求强联通分量+缩点

    提到Tarjan算法就不得不提一提Tarjan这位老人家 Robert Tarjan,计算机科学家,以LCA.强连通分量等算法闻名.他拥有丰富的商业工作经验,1985年开始任教于普林斯顿大学.Tarj ...

  7. USACO06JAN The Cow Prom /// tarjan求强联通分量 oj24219

    题目大意: n个点 m条边的图 求大小大于1的强联通分量的个数 https://www.cnblogs.com/stxy-ferryman/p/7779347.html tarjan求完强联通分量并染 ...

  8. tarjan求强联通分量 模板

    void tarjan(int u) { dfn[u]=low[u]=++dfs_clock; stack_push(u); for (int c=head[u];c;c=nxt[c]) { int ...

  9. 培训补坑(day2:割点与桥+强联通分量)

    补坑ing... 好吧,这是第二天. 这一天我们主要围绕的就是一个人:tarjan......创造的强联通分量算法 对于这一天的内容我不按照顺序来讲,我们先讲一讲强联通分量,然后再讲割点与桥会便于理解 ...

随机推荐

  1. 洛谷—— P1097 统计数字

    https://www.luogu.org/problem/show?pid=1097 题目描述 某次科研调查时得到了n个自然数,每个数均不超过1500000000(1.5*10^9).已知不相同的数 ...

  2. POJ 1155

    很久以前做的树形DP题,今天再遇到时,竟然不会了,所以写写.. 设数组: prf[MAX][MAX],cost[MAX],sum[MAX].分别表示,在第i个结点为根的子树内的情况下,若有j个用户申请 ...

  3. 【cocos2d-x 3.7 飞机大战】 决战南海I (十二) 游戏结束场景

    游戏结束的时候,要显示分数.还要可以选择是返回主场景还是退出游戏 // 退出游戏 void menuCloseCallback(cocos2d::Ref* pSender); // 返回主界面 voi ...

  4. Xamarin Mono For Android、Monotouch 安装

    一.Windows下面的安装 1. 安装环境介绍:    Win8.1 企业版64位或Win7 64.VS2013 update4 2. 安装jdk    到oracle官方下载jdk-8u45-wi ...

  5. luogu1403 约数研究

    题目大意:给出n,求1~n所有数的约数个数的和. 将“1~n所有数的约数”的模板中的factor[i*j].push_back(i)改为FactorCnt[i*j]++,最后再求一次和即可. #inc ...

  6. EOJ 2844 排序去重

    有 n 个 1 到 1000 之间的整数 (1≤n≤100),对于其中重复的数字,只保留一个,把其余相同的数去掉.然后再按照指定的排序方式把这些数排序. Input 第 1 行为字母 A 或 D,A ...

  7. xss 记录cookie

    <p> <img src="http://act.ci123.com/global/ueditor_new/php/upload/98591403834900.jpg&qu ...

  8. Vue2.0框架搭建基础操作及目录说明

    一.概述 vue.js是一套构建用户界面的渐进式框架.vue采用自底向上增量开发的设计.vue的核心库只关心视图层,非常容易学习,非常容易与其它库和已有项目整合.vue完全有能力驱动采用单文件组件和v ...

  9. eclipse中文汉字看不清或过小的问题解决方法!!

    把字体修改为 中欧字体就可以看清汉字

  10. 【转】window 安装redis服务、卸载redis服务和启动redis服务

    1.安装redis服务 redis-install.bat 1 echo install redis-server23 D:\redis\redis-server.exe --service-inst ...