tarjan图论算法

标签: tarjan 图论 模板


洛谷P3387 【模板】缩点



算法:Tarjan有向图强连通分量+缩点+DAGdp

代码:

#include <cstdio>
#include <cstring>
#include <vector>
#include <queue>
#include <algorithm>
#include <iostream>
#define psk push_back
using namespace std; const int MAXN = 1e5 + 50; int dfn[MAXN], low[MAXN], dfscnt = 0, scccnt = 0;
int sccnum[MAXN], s[MAXN], in[MAXN], top = 0;
int p0[MAXN], p[MAXN], d[MAXN]; vector<int> G[MAXN], G0[MAXN];
queue<int> q; inline int read()
{
int res = 0, f = 1;
char ch;
ch = getchar(); while(!isdigit(ch)){
if(ch == '-')
f = -1; ch = getchar();
} while(isdigit(ch)){
res = res * 10 + ch - 48; ch = getchar();
} return f * res;
}
void tarjan(int now)
{
dfn[now] = low[now] = ++ dfscnt;
s[top ++] = now; for(int i = 0; i < G0[now].size(); i ++){
int v = G0[now][i]; if(!dfn[v]){
tarjan(v);
low[now] = min(low[now], low[v]);
}
else if(!sccnum[v])
low[now] = min(low[now], dfn[v]);
} if(low[now] == dfn[now]){ scccnt ++; do{
sccnum[s[-- top]] = scccnt;
}while(s[top] != now); }
return;
} int topoo()
{ for(int i = 1; i <= scccnt; i ++)
if(!in[i]){
d[i] = p[i];
q.push(i); } while(!q.empty()){
int u = q.front();q.pop(); for(int i = 0; i < G[u].size(); i ++){
int v = G[u][i]; if(d[v] < d[u] + p[v])
d[v] = d[u] + p[v]; in[v] --; if(!in[v])
q.push(v);
}
} return *max_element(d + 1, d + 1 + scccnt);
} int main()
{
int n, m; n = read(), m = read(); for(int i = 1; i <= n; i ++)
p0[i] = read(); for(int i = 0; i < m; i ++){
int u, v; u = read(), v = read(); G0[u].psk(v);
} for(int i = 1; i <= n; i ++)
if(!dfn[i])
tarjan(i); for(int i = 1; i <= n; i ++){
p[sccnum[i]] += p0[i]; for(int j = 0; j < G0[i].size(); j ++){
int v = G0[i][j];
if(sccnum[i] == sccnum[v])
continue;
G[sccnum[i]].psk(sccnum[v]); in[sccnum[v]] ++;
}
} printf("%d", topoo()); return 0;
}

洛谷P3388 【模板】割点(割顶)

算法:tarjan求无向图割点割边

代码:

#include <cstdio>
#include <cstring>
#include <vector>
#include <iostream>
#define pbk push_back
using namespace std; const int MAXN = 1e5 + 50; int dfn[MAXN], low[MAXN], n, m;
int dfscnt = 0, iscut[MAXN]; vector<int> G[MAXN]; inline int read()
{
int res = 0, f = 1; char ch; ch = getchar(); while(!isdigit(ch)){
if(ch == '-')
f = -1; ch = getchar();
} while(isdigit(ch)){
res = (res << 3) + (res << 1) + ch - 48; ch = getchar();
} return f * res;
}
void tarjan(int now, int rt)
{
int chcnt = 0; dfn[now] = low[now] = ++ dfscnt; for(int i = 0; i < G[now].size(); i ++){
int v = G[now][i]; if(!dfn[v]){
tarjan(v, rt);
low[now] = min(low[now], low[v]); if(now == rt)
chcnt ++;
else if(low[v] >= dfn[now])
iscut[now] = 1;
} else
low[now] = min(low[now], dfn[v]);
} if(chcnt >= 2)
iscut[now] = 1;
return;
} int main()
{
int n, m, tot = 0; n = read(), m = read(); for(int i = 0; i < m; i ++){
int u, v; u = read(), v = read(); G[u].pbk(v);
G[v].pbk(u);
} for(int i = 1; i <= n; i ++)
if(!dfn[i])
tarjan(i, i); for(int i = 1; i <= n; i ++)
if(iscut[i])
tot ++;
printf("%d\n", tot);
for(int i = 1; i <= n; i ++)
if(iscut[i])
printf("%d ", i); return 0;
}

求无向图边双连通分量

#include <cstdio>
#include <cstring>
#include <vector>
#include <queue>
#include <algorithm>
#include <iostream>
#define pbk push_back
using namespace std; const int MAXN = 1e5 + 50; vector<int> G[MAXN], bcc[MAXN]; int low[MAXN], dfn[MAXN], bnum[MAXN], s[MAXN];
int n, m, top = 0, dfscnt = 0, bcnt = 0; inline int read()
{
int res = 0, f = 1;
char ch; ch = getchar(); while(!isdigit(ch)){
if(ch == '-')
f = -1; ch = getchar();
} while(isdigit(ch)){
res = (res << 3) + (res << 1) + ch - 48; ch = getchar();
} return f * res;
}
void tarjan(int now, int fa)
{
dfn[now] = low[now] = ++ bcnt;
s[top ++] = now; int flag = 0;
for(int i = 0; i < G[now].size(); i ++){
int v = G[now][i]; if(v == fa && !flag){
flag = 1;
continue;
} if(!dfn[v]){
tarjan(v, now);
low[now] = min(low[now], low[v]);
}
else if(!bnum[v])
low[now] = min(low[now], dfn[v]);
} if(low[now] == dfn[now]){
bcnt ++; do{
bnum[s[-- top]] = bcnt;
bcc[bcnt].pbk(s[top]);
}while(s[top] != now);
} return ;
}
int main()
{
n = read(), m = read(); for(int i = 0; i < m; i ++){
int u, v; u = read(), v = read(); G[u].pbk(v);
G[v].pbk(u);
} for(int i = 1; i <= n; i ++)
if(!dfn[i])
tarjan(i, 0); printf("%d\n", bcnt); for(int i = 1; i <= bcnt; i ++){ printf("%d ", i); for(int j = 0; j < bcc[i].size(); j ++)
printf("%d ", bcc[i][j]);
printf("\n");
}
return 0;
}

求无向图点双连通分量

#include <cstdio>
#include <cstring>
#include <vector>
#include <queue>
#include <algorithm>
#include <iostream>
#define pbk push_back
using namespace std; const int MAXN = 1e5 + 50; int low[MAXN], dfn[MAXN], n, m;
int s[MAXN], top = 0, bcnt = 0, dfscnt = 0; vector<int> G[MAXN], bcc[MAXN]; inline int read()
{
int res = 0, f = 1; char ch; ch = getchar(); while(!isdigit(ch)){
if(ch == '-')
f = -1;
ch = getchar();
} while(isdigit(ch)){
res = (res << 3) + (res << 1) + ch - 48; ch = getchar();
} return f * res;
}
void tarjan(int now, int rt)
{
low[now] = dfn[now] = ++ bcnt; s[top ++] = now;
if(now == rt && !G[now].size()){
bcc[++ bcnt].pbk(s[-- top]);
return ;
}
for(int i = 0; i < G[now].size(); i ++){
int v = G[now][i]; if(!dfn[v]){
tarjan(v, rt);
low[now] = min(low[now], low[v]); if(low[v] >= dfn[now]){
do{
bcnt ++; bcc[bcnt].pbk(s[--top]);
}while(s[top] != v); bcc[bcnt].pbk(now);
}
}
else
low[now] = min(low[now], dfn[v]);
}
return; } int main()
{
n = read(), m = read(); for(int i = 0; i < m; i ++){
int u, v;
u = read(), v = read(); G[u].pbk(v);
G[v].pbk(u);
} for(int i = 1; i <= n; i ++){
if(!dfn[i])
tarjan(i, i); } for(int i = 1; i <= bcnt; i ++){
printf("%d ", i);
for(int j = 0; j < bcc[i].size(); j ++)
printf("%d ", bcc[i][j]);
printf("\n");
} return 0;
}

tarjan图论算法的更多相关文章

  1. 图论算法-Tarjan模板 【缩点;割顶;双连通分量】

    图论算法-Tarjan模板 [缩点:割顶:双连通分量] 为小伙伴们总结的Tarjan三大算法 Tarjan缩点(求强连通分量) int n; int low[100010],dfn[100010]; ...

  2. [算法模版]Tarjan爷爷的几种图论算法

    [算法模版]Tarjan爷爷的几种图论算法 前言 Tarjan爷爷发明了很多图论算法,这些图论算法有很多相似之处(其中一个就是我都不会).这里会对这三种算法进行简单介绍. 定义 强连通(strongl ...

  3. NOIp 图论算法专题总结 (2)

    系列索引: NOIp 图论算法专题总结 (1) NOIp 图论算法专题总结 (2) NOIp 图论算法专题总结 (3) 树链剖分 https://oi-wiki.org/graph/heavy-lig ...

  4. LCA问题的ST,tarjan离线算法解法

    一  ST算法与LCA 介绍 第一次算法笔记这样的东西,以前学算法只是笔上画画写写,理解了下,刷几道题,其实都没深入理解,以后遇到新的算法要把自己的理解想法写下来,方便日后回顾嘛>=< R ...

  5. 图论算法-最小费用最大流模板【EK;Dinic】

    图论算法-最小费用最大流模板[EK;Dinic] EK模板 const int inf=1000000000; int n,m,s,t; struct node{int v,w,c;}; vector ...

  6. 图论算法-网络最大流【EK;Dinic】

    图论算法-网络最大流模板[EK;Dinic] EK模板 每次找出增广后残量网络中的最小残量增加流量 const int inf=1e9; int n,m,s,t; struct node{int v, ...

  7. LCA最近公共祖先(Tarjan离线算法)

    这篇博客对Tarjan算法的原理和过程模拟的很详细. 转载大佬的博客https://www.cnblogs.com/JVxie/p/4854719.html 第二次更新,之前转载的博客虽然胜在详细,但 ...

  8. LCA(最近公共祖先)--tarjan离线算法 hdu 2586

    HDU 2586 How far away ? Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/ ...

  9. 最近公共祖先LCA Tarjan 离线算法

    [简介] 解决LCA问题的Tarjan算法利用并查集在一次DFS(深度优先遍历)中完成所有询问.换句话说,要所有询问都读入后才开始计算,所以是一种离线的算法. [原理] 先来看这样一个性质:当两个节点 ...

随机推荐

  1. Python-xlwt库的基本使用

    安装xlwt库 pip install xlwt 基本使用 ①创建工作簿 wa = xlwt.Workbook() ②添加工作表 添加“class”工作表 b = wa.add_sheet('clas ...

  2. layui 滚动加载

    直接上核心代码,其实官网介绍的很详细: var pageSize = 5;//每次请求新闻的条数 flow.load({ elem: '#newsList' //指定列表容器 ,scrollElem: ...

  3. vue render函数解析

    一.render 函数的作用: 写一些vue.js的template太繁琐,利用render,可以使用js来生成模板,更加灵活和简便. 二.使用render前提: 官网也说了.在深入渲染函数之前推荐阅 ...

  4. 发送RCS成功的消息log_1

    //12-02 16:39:00.869323 24174 27394 I CarrierServices: [1172] cpb.x: Send INVITE//12-02 16:39:00.920 ...

  5. 入职小白随笔之Android四大组件——广播详解(broadcast)

    Broadcast 广播机制简介 Android中的广播主要可以分为两种类型:标准广播和有序广播. 标准广播:是一种完全异步执行的广播,在广播发出之后,所有的广播接收器几乎都会在同一时刻接收到这条广播 ...

  6. IP安全,DDoS攻击、tearDrop攻击和微小IP碎片攻击

    目录 arp安全 IP报文格式 DoS攻击 tear drop攻击 微小碎片攻击 IP欺骗,留后门 arp安全 以太网帧的type =0806 表示arp arp攻击:hack伪造arp应答包给tar ...

  7. python中 遇到的读取坑2.7和3.6版本的问题

    2.7读取,需要使用io.open 3.x使用open 使用io.open的时候路径需要使用\\ 目前io.open的文件名不能为中文

  8. C# List、Array、Dictionary之间相互转换

    Array转换为List List转换为Array Array转Dictionary Dictionary转Array List转Dictionary Dictionary转List IQueryab ...

  9. 12.2 中的Data Guard Standby 密码文件自动同步 (Doc ID 2307365.1)

    Data Guard Standby Automatic Password file Synchronization in 12.2 (Doc ID 2307365.1) APPLIES TO: Or ...

  10. 【bzoj4671】异或图(容斥+斯特林反演+线性基)

    传送门 题意: 给出\(s,s\leq 60\)张图,每张图都有\(n,n\leq 10\)个点. 现在问有多少个图的子集,满足这些图的边"异或"起来后,这张图为连通图. 思路: ...