割点与割边

概念:

割点是无向联通图中的一个特殊的点,删去中这个点后,此图不再联通,而所以满足这个条件的点所构成的集合即为割点集合。
除了割点还有一种问题是求割边(也称桥),即在一个无向图中删除某条边后,图不再连通。

如何求割点?

很容易想到的方法是:依次删除每一个顶点,然后用dfs或者bfs来检查图是否依然连通。如果删除某个顶点后,导致图不再连通,那么刚才删除的顶点就是割点。

这种方法的时间复杂度是O(N(N+M))。

下面寻找复杂度低的方法来解决。

首先从图中任意节点开始dfs遍历上图,得到一颗生成树, 如下图,圆圈中数字是顶点编号, 圆圈右上角的数表示这个顶点在遍历时是第几个被访问到的,叫做“时间戳”。

在遍历的时候一定会遇到割点, 关键是如何认定一个顶点是割点呢?

在深度优先遍历的时候访问到u(点2)点,此时图被u (点2)分割成两部分,一部分为已访问点,另一部分为未访问点。如果u是割点,那剩下的未被访问点中至少有一个点在不经过点k的情况下无论如何也回不到已访问的点。

基本思路

假如到了u后,图中还有顶点v是没有访问过的点,如何判断v在不经过u的情况下是否还能回到之前访问过的任意一个点?u是v的父亲,而之前访问过的顶点就是祖先。 也就是如何检测v在不经过父亲u的情 况下还能否回到祖先。那就是对v再 进行一次dfs,但此次遍历不经过u, 看能否回到祖先。不能u即为割点。

如何求割边?

只需将求割点的算法修改一个符号就可以。 只需将low[v]>=num[u]改为low[v]>num[u]。

这是为什么呢?

low[v]和num[u]相等则表示还可以回到父亲结点; 而low[v]>num[u]则表示连父亲都回不到了。倘若顶点v不能回到祖先,也没有另外的路能回到父亲,那么 u-v 这条边就是割边。

割点代码

#include<bits/stdc++.h>
#define ll long long
using namespace std;
const int N = 1e4 + 10;
const int M = 1e5 + 10;
struct edge{
    int to,net;
}e[M * 2];
    int head[N], dfn[N],low[N];
    int idx = 0;
    int cnt = 0;
    void add(int u,int v){ //建图
        e[++cnt].net = head[u];
        e[cnt].to = v;
        head[u] = cnt;
    }
    void tarjan(int u,int fa){ // 无向图的割点
        dfn[u] = low[u] = ++idx;
        int cnt = 0;
        for(int i = head[u];i != 0;i = e[i].net){
            int v = e[i].to;
            if(v == fa) continue;
            if(!dfn[v]){ // 没有被遍历过
                tarjan(v,u);
                low[u] = min(low[u],low[v]);
                if(low[v] >= dfn[u]){
                    cnt ++;
                    if(u != root || cnt > 1)
                        cout << u << "是割点" << endl;
                    }
                }
                else{ // 能去的这个点已经被搜过了
                    low[u] = min(low[u],dfn[v]);
                }
            }
        }
int main(){
    ios::sync_with_stdio(false);cin.tie(0);
    return 0;
}

求赞😭

  • 3
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
拓扑排序、割点割边以及强连通分量是图论中的重要概念和算法。 1. 拓扑排序(Topological Sorting): 拓扑排序是对有向无环图(DAG)进行排序的一种算法。拓扑排序可以得到一个顶点的线性序列,使得对于任意一条有向边(u, v),在序列中顶点u都排在顶点v的前面。拓扑排序常用于表示任务之间的依赖关系,例如在工程项目中确定任务的执行顺序。 2. 割点割边(Cut Vertex and Cut Edge): 割点是指在无向连通图中,如果移除该顶点以及与该顶点相连的所有边,会导致图不再连通,则该顶点被称为割点割边是指在无向连通图中,如果移除该边,会导致图不再连通,则该边被称为割边割点割边的存在可以影响到图的连通性,因此在网络设计、通信等领域有着重要的应用。 3. 强连通分量(Strongly Connected Component): 强连通分量是指在有向图中,如果对于图中任意两个顶点u和v,存在从u到v和从v到u的路径,那么称u和v在同一个强连通分量中。强连通分量可以将有向图的顶点划分成若干个子集,每个子集内的顶点之间互相可达。强连通分量可以用于分析网络中的关键节点,寻找网络的可靠性,以及在编译器设计中进行代码优化等领域。 这些概念和算法在图论中都有着广泛的应用,并且还有许多相关的算法和扩展。深入理解和掌握这些概念和算法,可以帮助我们更好地理解和解决各种与图相关的问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值