何为割点?也就是题目中的关键点。在一个无向图中,去掉一个点,这个无向图会变成多个子图,那么这个点就叫做割点
同理,割边也是如此,如果去掉一条边,能让无向图变成多个子图,那么这条边叫做割边,所谓的桥。
那么tarjan是如何求的割点的呢?
如果u为割点,当且仅当满足下面的1/2
1、如果u为树根,那么u必须有多于1棵子树
2、如果u不为树根,那么(u,v)为树枝边,当Low[v]>=DFN[u]时。
割点的求法倒是看明白了,条件1的意思是若为根,下面如果只有一颗子树,也就是整个图是强连通,那么去掉根节点,肯定不会变成多个子图,因此也不会成为割点。只有大于一颗子树,去掉根节点,才会有两个或者2个以上的子图,从而才能成为割点
条件2也比较好理解,u不为树根,那么u肯定有祖先,如果存在Low【v】>=DFN【u】时,表示u的子孙只能通过u才能访问u的祖先,这也就是说,不通过u,u的子孙无法访问u的祖先,那么如果去掉u这个节点,就会至少分为两个子图,一个是u祖先,一个是u子孙的。
#include <stdio.h> #include <stdlib.h> #include <string.h> #define nMax 110 #define Min(a,b) (a<b?a:b) #define Max(a,b) (a>b?a:b) int map[nMax][nMax]; int DFN[nMax],Low[nMax]; bool isVisted[nMax]; int gPoint[nMax]; int index, root; int n,ans; void tarjan(int u) { DFN[u] = Low[u] = ++index; isVisted[u] = true; for (int i = 1; i <= n; ++ i) { if (map[u][i]) { if (!isVisted[i]) { tarjan(i); Low[u] = Min(Low[u], Low[i]); if (Low[i] >= DFN[u] && u != 1)//if it is not root { gPoint[u] ++; //删除u点后,分成的连通分量个数 } else if (u == 1)//if it is root { root ++; } } else { Low[u] = Min(Low[u], DFN[i]); } } } } int main() { while (scanf("%d", &n) && n) { int u, v; memset(map, 0, sizeof(map)); memset(isVisted, false, sizeof(isVisted)); memset(gPoint, 0, sizeof(gPoint)); ans = root = index = 0; while (scanf("%d", &u) && u) { while (getchar() != '\n') { scanf("%d", &v); map[u][v] = 1; map[v][u] = 1; } } tarjan(1); if (root > 1) { ans ++; } for (int i = 2; i <= n; ++ i) { if (gPoint[i]) { ans ++; } } printf("%d\n", ans); } return 0; }