当前位置: 首页 > article >正文

图论:Tarjan算法的使用,找连通分量、割点、桥

感谢B站:“邋遢大哥233”的教学视频

Tarjan算法

Tarjan算法是处理有关图的连通性问题的算法。下面是有关图的连通性的一些基本概念:

图的连通性

连通图(Connected Graph):无向图中任意两个顶点都是连通的,即存在一条路径连接它们.
强连通图(Strongly Connected Graph):有向图中任意两个顶点都是强连通的,即存在从一个顶点到另一个顶点的路径,反之亦然.
连通分量(Connected Component):无向图中的最大连通子图.
强连通分量(Strongly Connected Component):有向图中的最大强连通子图.
割点(Cut Vertex):删除后会使图的连通性降低的顶点.如果去除某个顶点及与其相关的边后,连通分量的数量增加,则该点为割点。
桥(Bridge):删除后会使图的连通性降低的边.如果去除某条边后,连通分量的数量增加,则该点为桥。
在有向图中tarjan算法可以求连通分量,而在无向图中它可以求割点和桥。

个人对算法的理解

tarjan算法通过找强连通分量中的环的方法来求强连通分量。它通过深度遍历按遍历的顺序给每个顶点标上一个时间戳。如果某个顶点U是一个强连通分量中的顶点,那么深度遍历它的相邻顶点。在这次遍历的顶点中,必然有一些顶点它们的邻接顶点的时间戳要小于等于U的时间戳。因为在强连通分量中,从U出发必然会回到U。
所以,用两个数组dfn和low分别记录当前顶点的时间和从给点出发所能达到的最早的点的时间。如果dfn[u] == low[u] 说明不能回到更早的顶点;如果dfn[u] < low[u] 说明可以回到更早的顶点,存在强连通分量。
所以算法的关键在于时间戳,通过判断是否能回到过去,来判断是否存在环。把空间问题转化为了时间问题,很巧妙的设计。
顺便说一下,算法的时间复杂度与图的表示有关,如果用邻接矩阵,时间是:O(V^2);如果用邻接表,时间是:O(V+E)。V是顶点数,E是边数.

代码

本题以leetcode1192 查找集群内的关键连接 为背景,原题目是求无向图的桥。

求有向图强连通分量

在这里插入图片描述

class Solution {
    vector<int> dfn, low, inStack; // 记录当前时间,可以达到的最早时间,和标记是否已遍历
    stack<int> S;  // 记录节点
    int step = 1;  // 记录时间
    vector<vector<int>> graph, sccMap; // 分别邻接表, 记录连通分量
    void tarjan(int u){
        dfn[u] = low[u] = step ++;
        S.push(u);
        inStack[u] = 1; // 标记时间和入栈

        for(int v : graph[u]){
            if(dfn[v] == 0) tarjan(v); // 深度遍历为遍历的点
            if(inStack[v]) low[u] = min(low[u], low[v]);  // 记录可以到达的最早时间
        }

        if(dfn[u] == low[u]){ // 不能到达更早的点,说明是一个强连通分量的根
            int v;
            vector<int> scc; // 记录连通分量
            do{
                v = S.top();
                S.pop();
                //scc.push_back(v);
            }while(v != u);
            sccMap.push_back(scc);
        }
    }

public:
    vector<vector<int>> criticalConnections(int n, vector<vector<int>>& connections) {
        dfn.resize(n, 0);
        low.resize(n, 0);
        inStack.resize(n, 0);
        graph.resize(n);

        for(const auto &edge: connections){// 假设这是有向图
            graph[edge[0]].push_back(edge[1]);
        }

        tarjan(0);
        return sccMap;
    }
};

求无向图的割点

如果当节点不为根,且子节点不能到达比当前顶点更早的节点,去掉前期节点,图必有早于当前节点和晚于当前节点的两部分。
如果当前节点有两个子节点去掉当前节点,两个子节点不连通

割点和桥

class Solution {
    vector<int> dfn, low, inStack, cuts; // 记录当前时间,可以达到的最早时间,标记是否已遍历和割点
    stack<int> S;
    int step = 1;  // 时间
    vector<vector<int>> graph; // 邻接表
    void tarjan(int u, int f){
        dfn[u] = low[u] = step ++;
        S.push(u);
        inStack[u] = 1; // 标记时间和入栈
		
		int coutSon = 0;
        for(int v : graph[u]){
            if(v == f) continue; // 避免回到父节点
            if(dfn[v] == 0){
            	tarjan(v, u);
            	countSon ++
            }
            if(inStack[v]) low[u] = min(low[u], low[v]);
        }
        if(countSon > 1){
        	cuts.push_back(u);
        	return;
        }
        if(dfn[u] == 1) return;
        for(int v : graph[u]){
        	if(dfn[u] <= low[v]) cuts.push_back(v);
        }
    }

public:
    vector<vector<int>> criticalConnections(int n, vector<vector<int>>& connections) {
        dfn.resize(n, 0);
        low.resize(n, 0);
        inStack.resize(n, 0);
        graph.resize(n);

        for(const auto &edge: connections){
            graph[edge[0]].push_back(edge[1]);
            graph[edge[1]].push_back(edge[0]);
        }

        tarjan(0, -1);
        return cuts;
    }
};

求无向图的桥

当边的下一个节点无法到达,比边的上一个更早的节点,该边为桥。
class Solution {
    vector<int> dfn, low, inStack;
    stack<int> S;
    int step = 1;
    vector<vector<int>> ans, graph;
    void tarjan(int u, int f){
        dfn[u] = low[u] = step ++;
        S.push(u);
        inStack[u] = 1;

        for(int v : graph[u]){
            if(v == f) continue;
            if(dfn[v] == 0) tarjan(v, u);
            if(inStack[v]) low[u] = min(low[u], low[v]);
            if(dfn[u] < low[v]) ans.push_back({u, v});
        }
    }

public:
    vector<vector<int>> criticalConnections(int n, vector<vector<int>>& connections) {
        dfn.resize(n, 0);
        low.resize(n, 0);
        inStack.resize(n, 0);
        graph.resize(n);

        for(const auto &edge: connections){
            graph[edge[0]].push_back(edge[1]);
            graph[edge[1]].push_back(edge[0]);
        }

        tarjan(0, -1);
        return ans;
    }
};

http://www.kler.cn/a/472146.html

相关文章:

  • spring mvc源码学习笔记之九
  • 大模型LLM-Prompt-CRISPE
  • 关于Mac使用VSCode连接虚拟机
  • Three.js - 打开Web 3D世界的大门
  • thinkphp6.0常用设计模式实例
  • unity 播放 序列帧图片 动画
  • oracle 数据库回收站恢复误删除表
  • Elixir语言的字符串处理
  • Elixir语言的多线程编程
  • Android Audio基础(53)——PCM逻辑设备Write数据
  • 让你的网页动起来:深入理解 CSS 动画和过渡
  • 红日靶场12457-2024
  • 【flink-cdc】flink-cdc 3版本debug启动pipeline任务,mysql-doris
  • 【马来西亚理工大学主办,ACM出版】2025年大数据、通信技术与计算机应用国际学术会议(BDCTA 2025)
  • Python3刷算法来呀,贪心系列题单
  • 大数据-234 离线数仓 - 异构数据源 DataX 将数据 从 HDFS 到 MySQL
  • SQL编程语言
  • pytorch 比较两个张量的是否相等的函数介绍
  • Python爬虫应用领域
  • 计算机网络:虚拟机虚拟网络配置
  • 鸿蒙中黑白版
  • 基于RedHat9部署WordPress+WooCommerce架设购物网站
  • SQL Server存储过程来实现分页功能
  • TRELLIS - 生成 3D 作品的开源模型
  • KUKA机器人如何修改程序并下载到机器人控制器中?
  • uniapp uni-popup使用scroll-view滚动时,底部按钮设置position:fixed失效,部分ios设置有问题