poj 1966 Cable TV Network

给出一个无向图,求该图的点连通度。

点连通度:最小割点集合中的顶点数。

0<=n<=50.

最小割。

边连通度很好求,只需要随便固定源点枚举汇点求最小割取min即可。

点连通度采用点边转化的思想,将每个点拆点,入点与出点连边将点上的信息反映在边上。

每个点的入点向出点连一条容量为1的边。

每条边拆成两条有向边,对于有向边(u,v)从u的出点向v的入点连一条容量为正无穷的边。

n2枚举源点和汇点,最小割取min即为答案。

割掉入出点间的边相当于删掉该点。真正的边不能删掉,设成正无穷防割。

点连通度不能固定源点枚举汇点,因为源汇两点默认不在最小割点集合中,所以必须枚举源汇。

 1 #include<iostream>
 2 #include<cstring>
 3 #include<cstdio>
 4 #include<algorithm>
 5 #include<queue>
 6 using namespace std;
 7 const int dian=105;
 8 const int bian=20005;
 9 const int INF=0x3f3f3f3f;
10 const int inf=100000;
11 int h[dian],ver[bian],val[bian],nxt[bian],ch[dian],cr[dian];
12 int n,m,tot,aa,bb,ans;
13 int S,T;
14 void add(int a,int b,int c){
15     tot++;ver[tot]=b;val[tot]=c;nxt[tot]=h[a];h[a]=tot;
16     tot++;ver[tot]=a;val[tot]=0;nxt[tot]=h[b];h[b]=tot;
17 }
18 bool tell(){
19     memset(ch,-1,sizeof(ch));
20     queue<int>q;
21     q.push(S);
22     ch[S]=0;
23     while(!q.empty()){
24         int t=q.front();
25         q.pop();
26         for(int i=h[t];i;i=nxt[i])
27             if(ch[ver[i]]==-1&&val[i]){
28                 ch[ver[i]]=ch[t]+1;
29                 q.push(ver[i]);
30             }
31     }
32     return ch[T]!=-1;
33 }
34 int zeng(int a,int b){
35     if(a==T)
36         return b;
37     int r=0;
38     for(int i=cr[a];i&&b>r;i=nxt[i])
39         if(ch[ver[i]]==ch[a]+1&&val[i]){
40             int t=zeng(ver[i],min(b-r,val[i]));
41             val[i]-=t,r+=t,val[i^1]+=t;
42             if(val[i])
43                 cr[a]=i;
44         }
45     if(!r)
46         ch[a]=-1;
47     return r;
48 }
49 int dinic(){
50     int r=0,t;
51     while(tell()){
52         for(int i=1;i<=n+n;i++)
53             cr[i]=h[i];
54         while(t=zeng(S,INF))
55             r+=t;
56     }
57     return r;
58 }
59 int main(){
60     while(scanf("%d%d",&n,&m)!=EOF){
61         memset(h,0,sizeof(h));
62         memset(nxt,0,sizeof(nxt));
63         tot=1;
64         ans=INF;
65         for(int i=1;i<=n;i++)
66             add(i,i+n,1);
67         for(int i=1;i<=m;i++){
68             scanf(" (%d,%d)",&aa,&bb);
69             add(aa+n+1,bb+1,inf);
70             add(bb+n+1,aa+1,inf);
71         }
72         for(int i=1;i<=n;i++)
73             for(int j=1;j<=n;j++){
74                 if(i==j)
75                     continue;
76                 S=i+n,T=j;
77                 ans=min(ans,dinic());
78                 for(int k=2;k<=tot;k+=2){
79                     val[k]+=val[k^1];
80                     val[k^1]=0;
81                 }
82             }
83         if(ans>=inf)
84             ans=n;
85         printf("%d\n",ans);
86     }
87     return 0;
88 }

 

posted @ 2016-12-28 14:08  dugudashen  阅读(326)  评论(0编辑  收藏  举报