Book---强连通分量

这几天一直在做强连通,现在总结一小下

1.定义

在一个有向图中,如果任意的两个点都是相互可达的,就说这个图是强连通的,有向图的极大强连通子图,称为强连通分量

2.求法

学的是白书上的tarjan算法

用到了DFS的时间戳

假设一个强连通分量C,其中的第一个点是 P,那么DFS下去,就一定能够找到一个K点,返回P点,这条DFS路径上的点就处于这个强连通分量C中

假如现在发现节点v,同时发现节点v最远只能够到达节点u,那么节点u就是这个强连通分量最先被发现的节点

这样就转化成了求一个点u最远能够到达的的祖先的d值

所以要定义一个low[]数组为节点u最早能够追溯到的祖先的pre[v]的值,pre[v]为节点v的时间戳

void dfs(int u){
    low[u] = pre[u] = ++dfs_clock;
    S.push(u);
    for(int i = first[u];~i;i = e[i].next){
        int v = e[i].v;
        if(!pre[v]){//子节点还没有被搜过,则搜下去,回溯回来更新low值
            dfs(v);
            low[u] = min(low[u],low[v]);
        }
        else if(!sc[v]) low[u] = min(low[u],pre[v]);//如果当前节点不属于其他scc,且已经被访问过,说明它是当前节点的祖先节点,直接用它的low更新
    }
    if(pre[u] == low[u]){
        scnt++;
        for(;;){
            int x = S.top();S.pop();
            sc[x] = scnt;
           // scn[scnt]++;这里可以统计一个连通分量里面有多少个点,或者当点有点权的时候,算出这个连通分量的权值
            if(x == u) break;
        }
    }
}

void find_scc(){
    while(!S.empty()) S.pop();
    scnt = dfs_clock = 0;
    memset(low,0,sizeof(low));memset(pre,0,sizeof(pre));
    memset(sc,0,sizeof(sc));memset(scn,0,sizeof(scn));

    for(int i = 1;i <= n;i++) if(!pre[i]) dfs(i);
}

3.现在做到的一些题目

现在做到的题目还是挺有限的,把自己做的总结下---以后遇到再补

(1)判断是否是一个强连通图

1)直接给一个有向图,判断是不是强连通的

hdu 1269

2) 附加一点限制的

加一个限制这个强连通分量里面的节点个数大于1的,tarjan算法里面直接再统计一下个数

poj 3180

( 2 )和入度出度有关的

1)加入几条边使得整个图强连通

白书上强连通那里的第一道例题

先tarjan,找出所有的强连通分量

再缩点,

再统计图的入度,出度,

另入度为0的有c1个,出度为0的有c2个,max(c1,c2)就是要求的

la 4287

hdu 2767

hdu 3836

这三道都是一样的

2)从哪些点出发,使得所有的点都能够走到

缩点,缩点后所有入度为0的就是要求的

hdu 1827 找入度为0的连通块,还附加了一个权值

poj 1236 max(c1,c2),和输出入度为0的连通分量的个数

poj 2553 统计出度为0的块

poj 2186 找有几个点是别的点都能够到达的,统计出度为0的块的个数c,

注意c > 1的情况,出度为0的块里面的点是相互不能够到达的,应该输出0

poj 2375 将图转化以后就是求max(c1,c2)

(3)和别的结合的

一般要用到缩点,缩完点之后形成一个有向无环图,构成一个DAG,

就可以dp,dfs,spfa

uva 11324

先求出所有强连通,再缩点,缩点完之后建立一个新的图

建图的办法是,扫一遍原来所有的边,如果发现这条边的两个端点u,v处在不同的连通分量中,就在sc[u],sc[v]之间连一条边

则转化成求DAG上权最大的路径

还可以有另一种办法

加一个源点进去,将这个源点与缩点完之后所有入度为0的点连一条边

再spfa求出权最大的路径

poj 3160和这题一样

poj 2762 缩点之后拓扑排序

poj 3114 缩点之后dijkstra,用floyd会t掉

poj 3592 图转化一下以后,spfa

现在就做了这些----

看的这两篇博客做得题目---

http://blog.csdn.net/accelerator_/article/details/40349411

http://blog.csdn.net/u013013910/article/category/2509879

时间: 2024-11-05 22:56:38

Book---强连通分量的相关文章

Kosaraju算法解析: 求解图的强连通分量

1. 定义 连通分量:在无向图中,即为连通子图. 上图中,总共有四个连通分量.顶点A.B.C.D构成了一个连通分量,顶点E构成了一个连通分量,顶点F,G和H,I分别构成了两个连通分量. 强连通分量:有向图中,尽可能多的若干顶点组成的子图中,这些顶点都是相互可到达的,则这些顶点成为一个强连通分量. 上图中有三个强连通分量,分别是a.b.e以及f.g和c.d.h. 2. 连通分量的求解方法 对于一个无向图的连通分量,从连通分量的任意一个顶点开始,进行一次DFS,一定能遍历这个连通分量的所有顶点.所以

POJ 2186 Popular Cows 强连通分量模板

题意 强连通分量,找独立的块 强连通分量裸题 #include <cstdio> #include <cstdlib> #include <cstring> #include <string> #include <algorithm> #include <iostream> using namespace std; const int maxn = 50005; int n, m; struct Edge { int v, next;

USACO network of school 强连通分量

这个题的意思是有一个有向图, 每个顶点可以发送软件到与其相连的顶点上, 现在问1,至少发送给几个顶点能满足所有顶点都收到软件, 2:如果想让这个图变成强连通图,至少添几条边.  特例是给定的图是一个强连通图的话答案是1, 0. 一般情况下我们先将这个图的强连通分量求出来缩成一个点然后统计入度为0的点和出度为0的点的个数, 答案一就是入度为0的点的个数, 答案就是他们两个之间的最大值.代码如下: /* ID: m1500293 LANG: C++ PROG: schlnet */ #include

强连通分量(学习心得)

定义:有向图强连通分量:在有向图G中,如果两个顶点vi,vj间(vi>vj)有一条从vi到vj的有向路径,同时还有一条从vj到vi的有向路径,则称两个顶点强连通如果有向图G的每两个顶点都强连通,称G是一个强连通图.有向图的极大强连通子图,称为强连通分量. 求强连通分量: vector<int>pic[maxn]; int dfn[maxn],low[maxn],ans[maxn]; bool ins[maxn]; stack<int>st; int dind=0,block=

POJ 2186:Popular Cows(强连通分量)

[题目链接] http://poj.org/problem?id=2186 [题目大意] 给出一张有向图,问能被所有点到达的点的数量 [题解] 我们发现能成为答案的,只有拓扑序最后的SCC中的所有点, 那么我们从其中一个点开始沿反图dfs,如果能访问到全图, 则答案为其所在SCC的大小,否则为0. [代码] #include <cstdio> #include <algorithm> #include <vector> #include <cstring>

【学习整理】Tarjan:强连通分量+割点+割边

Tarjan求强连通分量 在一个有向图中,如果某两点间都有互相到达的路径,那么称中两个点强联通,如果任意两点都强联通,那么称这个图为强联通图:一个有向图的极大强联通子图称为强联通分量.   算法可以在 的时间内求出一个图的所有强联通分量. 表示进入结点 的时间 表示从 所能追溯到的栈中点的最早时间 如果某个点 已经在栈中则更新  否则对 进行回溯,并在回溯后更新  #include<iostream> #include<cstdlib> #include<cstdio>

【强连通分量】tarjan算法及kosaraju算法+例题

阅读前请确保自己知道强连通分量是什么,本文不做赘述. Tarjan算法 一.算法简介 Tarjan算法是一种由Robert Tarjan提出的求有向图强连通分量的时间复杂度为O(n)的算法. 首先我们要知道两个概念:时间戳(DFN),节点能追溯到的最早的栈中节点的时间戳(LOW).顾名思义,DFN就是在搜索中某一节点被遍历到的次序号(dfs_num),LOW就是某一节点在栈中能追溯到的最早的父亲节点的搜索次序号. Tarjan算法是基于深度优先搜索的算法.在搜索过程中把没有Tarjan过的点入栈

有向图的强连通分量(tarjan算法)

强连通分量 有向图强连通分量:在有向图G中,如果两个顶点vi,vj间(vi>vj)有一条从vi到vj的有向路径,同时还有一条从vj到vi的有向路径,则称两个顶点强连通(strongly connected).如果有向图G的每两个顶点都强连通,称G是一个强连通图.有向图的极大强连通子图,称为强连通分量(strongly connected components). 考虑强连通分量C,设其中第一个被发现的点为x,则,C中其他的点都是x的后代.我们希望在x访问完成时立即输出C(可以同时记录C,输出代表

【BZOJ1051】1051: [HAOI2006]受欢迎的牛 tarjan求强连通分量+缩点

Description 每一头牛的愿望就是变成一头最受欢迎的牛.现在有N头牛,给你M对整数(A,B),表示牛A认为牛B受欢迎. 这种关系是具有传递性的,如果A认为B受欢迎,B认为C受欢迎,那么牛A也认为牛C受欢迎.你的任务是求出有多少头牛被所有的牛认为是受欢迎的. Input 第一行两个数N,M. 接下来M行,每行两个数A,B,意思是A认为B是受欢迎的(给出的信息有可能重复,即有可能出现多个A,B) Output 一个数,即有多少头牛被所有的牛认为是受欢迎的. Sample Input 3 3

POJ2186 Popular Cows 【强连通分量】+【Kosaraju】+【Tarjan】+【Garbow】

Popular Cows Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 23445   Accepted: 9605 Description Every cow's dream is to become the most popular cow in the herd. In a herd of N (1 <= N <= 10,000) cows, you are given up to M (1 <= M &l