求无向图的割点 (poj 1144 Network)

割点 :去掉该点后原来的图不连通(出现好几个连通分量),该点被称为割点。

注意删除某点意味着和该点关联的边也全部删除

求割点的伪代码

DFS(v1,father):
  dfn[v1] = low[v1] = ++dfsClock
  vis[v1] = true
  child = 0
  for each egde(v1,v2) in E:
    if(vis[v2] == false) : //(v1,v2)是父子边
        DFS(v2,v1)
       child++
        low[v1] = Min(low[v1],low[v2])
        if(v1 != root && low[v2] >= dfn[v1])
            v1 is cut point
        if(low[v2] > dfn[v1])
            edge(v1,v2) is bridge
    end if
    elseif(v2 != father && vis[v2] == true) : //v2已经被访问过,(v1,v2)是反向边
        low[v1] = Min(low[v1],dfn[v2])//此时要更新low[v1]
    end if

Init() :
  dfn[] = low[] = vis[] = 0
  dfsClock = 0

dfn【i】 记录 DFS树的深度,也就是,DFS中第几次遍历到的点,dfn的值就为几;

low【i】存储的是 i节点及i节点的后续节点通过反向边所能达到的最小的DFS深度

【更新low】

如果所处的边是树边,low【v1】=min(low【v1】,low【v2】)

如果所处的边是反向边,low【v1】=min(low【v1】,dfn【v2】)

【判断割点】

假设v2是v1的子节点

如果v1是根节点, 那么如果v1有两个或者两个以上的子节点,那么v1是割点

如果v1不是根节点,那么如果low【v2】 >=dfn【v1】,可以判断 v1是割点。

给个模板题 poj1144 network

题目链接:http://poj.org/problem?id=1144

【题目大意】

Telephone Line Company 在城市与城市之间建立了电话网络,有这样一种城市S, 如果S的电话服务崩溃了,那么 其他城市的电话并不能联通(就是至少有一个城市其他城市都连接不到)。

给你城市 与城市之间的关系, 找出有多少个 这样的城市 S;

典型的模板题, S就是图中的 割点

【源代码】

#include <iostream>
#include <vector>
#include <cstdio>
using namespace std;
const int maxn =110;
vector<int>G[maxn]; //用邻接链表存图
int dfn[maxn],low[maxn],dfsClock;
bool cutNode[maxn]; //记录是否为割点
int vis[maxn];
void DFS(int v1,int father){
	int child =0;
	vis[v1]=true;
	dfn[v1]=low[v1]= ++dfsClock;
	for(int i=0;i<G[v1].size();i++){
		int v2=G[v1][i];
		if(!vis[v2]){
			DFS(v2,v1);
			child++;
			low[v1] = min(low[v1],low[v2]);
			if(v1 != 1 && low[v2] >= dfn[v1]) //割点
				cutNode[v1]=true;
			if(v1 == 1 &&child >=2){ //如果是起点
				 cutNode[v1]=true;
			}
		}
		else if(vis[v2]&&v2!=father){ //如果是 反向边
			low[v1]=min(low[v1],dfn[v2]);
		}
	}
}
void init(){
	for(int i=0;i<maxn;i++){
		G[i].clear();
		dfn[i]=low[i]=0;
		cutNode[i]=0;
		vis[i]=0;
	}
	dfsClock = 0;
}
int main(){
	int n;
	while(scanf("%d",&n)!=EOF&&n){
		init();
		int tmp;char c;
		int v1,v2;
		while(scanf("%d",&v1)&&v1){
			while((c=getchar())!='\n'){
				scanf("%d",&v2);
				G[v1].push_back(v2);
				G[v2].push_back(v1);
			}
		}
		DFS(1,-1);
		int ans=0;
		for(int i=1;i<=n;i++){
			if(cutNode[i])
				ans++;
		}
		printf("%d\n",ans);
	}
	return 0;
}

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-12-28 18:41:59

求无向图的割点 (poj 1144 Network)的相关文章

Uva 315 求无向图的割点的个数

http://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem&problem=251 A Telephone Line Company (TLC) is establishing a new telephone cable network. They are connecting several places numbered by integers from 1 to 

求无向图的割点和桥

/** * 求 无向图的割点和桥 * 可以找出割点和桥,求删掉每个点后增加的连通块. * 需要注意重边的处理,可以先用矩阵存,再转邻接表,或者进行判重 * 调用solve输出割点数,全局变量bridge记录边的个数 */ #include <stdio.h> #include <string.h> #include <algorithm> #include <iostream> using namespace std; const int maxn=1001

POJ 1144 Network(无向图连通分量求割点)

题目地址:POJ 1144 求割点.推断一个点是否是割点有两种推断情况: 假设u为割点,当且仅当满足以下的1条 1.假设u为树根,那么u必须有多于1棵子树 2.假设u不为树根.那么(u,v)为树枝边.当Low[v]>=DFN[u]时. 然后依据这两句来找割点就能够了. 代码例如以下: #include <iostream> #include <cstdio> #include <string> #include <cstring> #include &

POJ 1144 Network(强连通分量求割点)

题目地址:POJ 1144 求割点.判断一个点是否是割点有两种判断情况: 如果u为割点,当且仅当满足下面的1条 1.如果u为树根,那么u必须有多于1棵子树 2.如果u不为树根,那么(u,v)为树枝边,当Low[v]>=DFN[u]时. 然后根据这两句来找割点就可以了. 代码如下: #include <iostream> #include <cstdio> #include <string> #include <cstring> #include <

图论(无向图的割顶):POJ 1144 Network

Network Description A Telephone Line Company (TLC) is establishing a new telephone cable network. They are connecting several places numbered by integers from 1 to N . No two places have the same number. The lines are bidirectional and always connect

tarjan算法--求无向图的割点和桥

一.基本概念 1.桥:是存在于无向图中的这样的一条边,如果去掉这一条边,那么整张无向图会分为两部分,这样的一条边称为桥无向连通图中,如果删除某边后,图变成不连通,则称该边为桥. 2.割点:无向连通图中,如果删除某点后,图变成不连通,则称该点为割点. 二:tarjan算法在求桥和割点中的应用 1.割点:1)当前节点为树根的时候,条件是“要有多余一棵子树”(如果这有一颗子树,去掉这个点也没有影响,如果有两颗子树,去掉这点,两颗子树就不连通了.) 2)当前节点U不是树根的时候,条件是“low[v]>=

poj 1144 Network 无向图求割点

Network Description A Telephone Line Company (TLC) is establishing a new telephone cable network. They are connecting several places numbered by integers from 1 to N . No two places have the same number. The lines are bidirectional and always connect

poj 1144 Network【双连通分量求割点总数】

Network Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 11042   Accepted: 5100 Description A Telephone Line Company (TLC) is establishing a new telephone cable network. They are connecting several places numbered by integers from 1 to N

poj 1144 Network【无向图求割顶模板题】

Description A Telephone Line Company (TLC) is establishing a new telephone cable network. They are connecting several places numbered by integers from 1 to N. No two places have the same number. The lines are bidirectional and always connect together