05-树8 File Transfer

  课上例题。

  这题考查对并查集的运用以及对之前并查算法的优化。

  1 #include <stdio.h>
  2 
  3 #define MAXN 10000                    /* 集合最大元素个数 */
  4 
  5 typedef int ElementType;            
  6 typedef ElementType SetName;        /* 默认用根节点的下标作为结点名称 */
  7 typedef ElementType SetType[MAXN];    /* 假设集合元素下标从 0 开始 */
  8 
  9 void Initialization(SetType S, int N);
 10 SetName Find(SetType S, ElementType X);
 11 void Union(SetType S, SetName Root1, SetName Root2);
 12 void Input_connection(SetType S);
 13 void Check_connection(SetType S);
 14 void Check_netWork(SetType S, int N);
 15 
 16 int main()
 17 {
 18     SetType S;
 19     int N;
 20     char in;
 21     scanf("%d\n", &N);
 22     Initialization( S, N );
 23     
 24     do {
 25         scanf("%c", &in);
 26         switch ( in ) {
 27             case 'I':    Input_connection( S );    break;
 28             case 'C':    Check_connection( S );    break;
 29             case 'S':    Check_netWork( S, N );    break;
 30         }
 31     } while ( in != 'S' );
 32     
 33     return 0;
 34 }
 35 
 36 void Input_connection(SetType S)
 37 {
 38     ElementType x1, x2;
 39     SetName Root1, Root2;
 40     
 41     scanf("%d %d\n", &x1, &x2);
 42     Root1 = Find(S, x1-1);
 43     Root2 = Find(S, x2-1);
 44     
 45      if ( Root1 != Root2 )
 46          Union(S, Root1, Root2);
 47 }
 48 
 49 void Check_connection(SetType S)
 50 {    /* 检查 是否 连通 */
 51     ElementType x1, x2;
 52     SetName Root1, Root2;
 53     
 54     scanf("%d %d\n", &x1, &x2);
 55     Root1 = Find(S, x1-1);
 56     Root2 = Find(S, x2-1);
 57     
 58     if ( Root1 == Root2 ) printf("yes\n");
 59     else                   printf("no\n");
 60     
 61 }
 62 
 63 void Check_netWork(SetType S, int N)
 64 {    /* 检查 连通性 */
 65     int i, cnt = 0;
 66     
 67     for ( i = 0; i < N; i++ ) {
 68         if ( S[i] < 0 )        cnt++; 
 69     }    
 70     
 71     if ( cnt == 1 )
 72         printf("The network is connected.\n");
 73     else 
 74         printf("There are %d components.\n", cnt);
 75 }
 76 
 77 void Union(SetType S, SetName Root1, SetName Root2)
 78 {
 79     /* 保证 小集合 并入 大集合 */
 80     if ( S[Root1] < S[Root2] ) { /* 如果 Root1 规模大 */
 81         S[Root1] += S[Root2];
 82         S[Root2] = Root1;         /* Root2 并入 Root1 */ 
 83     }
 84     else {                         /* 如果 Root2 规模大 */
 85         S[Root2] += S[Root1];
 86         S[Root1] = Root2;         /* Root1 并入 Root2 */ 
 87     }
 88 }
 89 
 90 SetName Find(SetType S, ElementType X)
 91 {
 92     if ( S[X] < 0 )  /* 找到集合的根 */ 
 93         return X;
 94     else
 95         return S[X] = Find(S, S[X]); /* 路径压缩 */ 
 96 }
 97 
 98 void Initialization(SetType S, int N)
 99 {    /* 集合元素全部初始化为 -1 */ 
100     int i;
101     for ( i = 0; i < N; i++ ) 
102         S[i] = -1;
103 }

  不得不说,这里的 switch 语句用的太漂亮了,赞。

posted @ 2018-11-21 21:02  望汐  阅读(147)  评论(0编辑  收藏  举报