图论算法-Tarjan模板 【缩点;割顶;双连通分量】
图论算法-Tarjan模板 【缩点;割顶;双连通分量】
为小伙伴们总结的Tarjan三大算法
Tarjan缩点(求强连通分量)
int n;
int low[100010],dfn[100010];
bool ins[100010];
int col[100010];//记录每个点所属强连通分量(即染色)
vector<int> map[100010];
stack<int> st;
int tot;//时间戳
int colnum;//记录强连通分量个数
void tarjan(int u)
{
low[u]=dfn[u]=++tot;
st.push(u);
ins[u]=true;
for(int j=0;j<map[u].size();j++)
{
int v=map[u][j];
if(!dfn[v])
{
tarjan(v);
low[u]=min(low[u],low[v]);
}
else if(ins[v])
low[u]=min(low[u],dfn[v]);
}
if(low[u]==dfn[u])
{
//到这里即发现了一个新的强连通分量
colnum++;
int temp;
int cont=0;
do
{
temp=st.top();
st.pop();
ins[temp]=false;
//将同一强连通分量的点染色,表示一个缩点
col[temp]=colnum;
//在这里也可以对该强连通分量进行一些其他操作
//例如保存该缩点所包含的原节点
}
while(temp!=u);
}
}
void init()
{
memset(dfn,0,sizeof(dfn));
memset(low,0,sizeof(low));
memset(col,0,sizeof(col));
memset(ins,false,sizeof(ins));
tot=colnum=0;
}
void solve()
{
init();
for(int i=1;i<=n;i++)
{
if(!dfn[i])
tarjan(i);
}
}
Tarjan求割点(割顶)
int n;
vector<int> map[100010];
int low[100010],dfn[100010];
bool cut[1000010];//记录割点
int tot;
void tarjan(int u,int fa)
{
low[u]=dfn[u]=++tot;
int child=0;
for(int j=0;j<map[u].size();j++)
{
int v=map[u][j];
if(!dfn[v])
{
child++;
tarjan(v,u);
low[u]=min(low[u],low[v]);
if(low[v]>=dfn[u])
cut[u]=true;
}
else if(dfn[v]<dfn[u]&&v!=fa)
low[u]=min(low[u],dfn[v]);
}
if(fa<0&&child==1)
cut[u]=false;
}
void init()
{
memset(dfn,0,sizeof(dfn));
memset(low,0,sizeof(low));
memset(cut,false,sizeof(cut));
tot=0;
}
void solve()
{
init();
for(int i=1;i<=n;i++)
{
if(!dfn[i])
tarjan(i,-1);
//**高亮**这里的第二个参数一定要设为负数
}
//cut[i]==true即表示i为割点
}
Tarjan求点-双连通分量
int n;
vector<int> map[100010];
int low[100010],dfn[100010];
bool cut[1000010];
int bcc[1000010];
int tot;
int bcc_cont//记录双连通分量个数;
struct edge{int u,v};
stack<edge> E;
void tarjan(int u,int fa)
{
low[u]=dfn[u]=++tot;
int child=0;
for(int j=0;j<map[u].size();j++)
{
int v=map[u][j];
edge e=(edge) {u,v};
if(!dfn[v])
{
E.push(e);
child++;
tarjan(v,u);
low[u]=min(low[u],low[v]);
if(low[v]>=dfn[u])
{
//到这里即发现了一个新的双连通分量
cut[u]=true;
bcc_cont++:
while(1)
{
edge temp=E.top();
E.pop();
if(bccno[temp.u]!=bcc_cont)
bccno[temp.u]=bcc_cont;
if(bccno[temp.v]!=bcc_cont)
bccno[temp.v]=bcc_cont;
if(temp.u==u&&temp.v==v)
break;
}
}
}
else if(dfn[v]<dfn[u]&&v!=fa)
{
E.push(e);
low[u]=min(low[u],dfn[v]);
}
}
if(fa<0&&child==1)
cut[u]=false;
}
void init()
{
memset(dfn,0,sizeof(dfn));
memset(low,0,sizeof(low));
memset(bccno,0,sizeof(bccno));
tot=bcc_cont=0;
}
void solve()
{
init();
for(int i=1;i<=n;i++)
{
if(!dfn[i])
tarjan(i,-1);//**高亮**这里的第二个参数一定要设为负数
}
}
其实三个算法思路都基本一致
毕竟都是同一个人提出的嘛
图论算法-Tarjan模板 【缩点;割顶;双连通分量】的更多相关文章
- Tarjan算法应用 (割点/桥/缩点/强连通分量/双连通分量/LCA(最近公共祖先)问题)(转载)
Tarjan算法应用 (割点/桥/缩点/强连通分量/双连通分量/LCA(最近公共祖先)问题)(转载) 转载自:http://hi.baidu.com/lydrainbowcat/blog/item/2 ...
- tarjan算法求无向图的桥、边双连通分量并缩点
// tarjan算法求无向图的桥.边双连通分量并缩点 #include<iostream> #include<cstdio> #include<cstring> ...
- Tarjan应用:求割点/桥/缩点/强连通分量/双连通分量/LCA(最近公共祖先)【转】【修改】
一.基本概念: 1.割点:若删掉某点后,原连通图分裂为多个子图,则称该点为割点. 2.割点集合:在一个无向连通图中,如果有一个顶点集合,删除这个顶点集合,以及这个集合中所有顶点相关联的边以后,原图变成 ...
- (转)Tarjan应用:求割点/桥/缩点/强连通分量/双连通分量/LCA(最近公共祖先)
基本概念: 1.割点:若删掉某点后,原连通图分裂为多个子图,则称该点为割点. 2.割点集合:在一个无向连通图中,如果有一个顶点集合,删除这个顶点集合,以及这个集合中所有顶点相关联的边以后,原图变成多个 ...
- tarjan求桥、割顶
若low[v]>dfn[u],则(u,v)为割边.但是实际处理时我们并不这样判断,因为有的图上可能有重边,这样不好处理.我们记录每条边的标号(一条无向边拆成的两条有向边标号相同),记录每个点的父 ...
- 图论--割边--Tarjan模板
#include<iostream> #include<stdio.h> #include<vector> using namespace std; const i ...
- 图论--割点--Tarjan模板
#include <iostream> #include <algorithm> #include <cstdio> #include <cstring> ...
- [Tarjan系列] Tarjan算法求无向图的双连通分量
这篇介绍如何用Tarjan算法求Double Connected Component,即双连通分量. 双联通分量包括点双连通分量v-DCC和边连通分量e-DCC. 若一张无向连通图不存在割点,则称它为 ...
- 点/边 双连通分量---Tarjan算法
运用Tarjan算法,求解图的点/边双连通分量. 1.点双连通分量[块] 割点可以存在多个块中,每个块包含当前节点u,分量以边的形式输出比较有意义. typedef struct{ //栈结点结构 保 ...
随机推荐
- 为什么选择.NETCore?
为什么.NETCore? 学习新的开发框架是一项巨大的投资.您需要学习如何在新框架中编写,构建,测试,部署和维护应用程序.作为开发人员,有许多框架可供选择,很难知道什么是最适合的工作.即使您正在使用. ...
- Java 线程和多线程执行过程分析
*/ .hljs { display: block; overflow-x: auto; padding: 0.5em; color: #333; background: #f8f8f8; } .hl ...
- 【转】GPS定位原理
一.距离测定原理 1.伪距测量 伪距测量是利用全球卫星定位系统进行导航定位的最基本的方法,其基本原理是:在某一瞬间利用GPS接收机同时测定至少四颗卫星的伪距,根据已知的卫星位置 和伪距观测值,采用距离 ...
- 基于 HTML5 WebGL 的 3D SCADA 主站系统
这个例子的初衷是模拟服务器与客户端的通信,我把整个需求简化变成了今天的这个例子.3D 的模拟一般需要鹰眼来辅助的,这样找产品以及整个空间的概括会比较明确,在这个例子中我也加了,这篇文章就算是我对这次项 ...
- Notepad++运行Java
插件NppExec使用 首先要让Notepad++编译和运行Java,前提是电脑里已经配置好了Java的环境 1,安装插件NppExec:解压出来提取NppExec.dll文件放在Notepad++安 ...
- 使用Filebeat和Logstash集中归档日志
方 案 Filebeat->Logstash->Files Filebeat->Redis->Logstash->Files Nxlog(Rsyslog.Logstash ...
- 系统uid在1-499的原因
1.因为是保留给系统使用的UID,为了与用户设置的账户区分,防止冲突. 2.并没有其他特别的意义, 3.也叫作虚拟用户,除了0之外,所有的UID在使用上并没有任何区别. 4.linux中文件和程序都要 ...
- sigmoid_cross_entropy_with_logits
sigmoid_cross_entropy_with_logits 原创文章,请勿转载!!! 函数定义 def sigmoid_cross_entropy_with_logits(_sentinel= ...
- 通过url获取相应的location信息
var properties = ['href', 'origin', 'host', 'hostname', 'port', 'pathname', 'search', 'hash']; var g ...
- 使用命令行生成jar包
测试用类 public class Hello { public static void main(String[] args) { System.out.println("hello wo ...