pku 1161 Walls

  1. #include <cstdio>
  2. #include <cstring>
  3. const int MAXN = 252 ;
  4. const int MAX_INF = 50000 ;
  5. int gMember[MAXN] ;
  6. int region[MAXN][MAXN] ;
  7. bool place[MAXN][MAXN] ;
  8. void Floyd ( int n )
  9. {
  10.     int i , j , k ;
  11.         
  12.     for ( k = 1 ; k <= n ; k++ )
  13.     {
  14.         for ( i = 1 ; i <= n ; i++ )
  15.             for ( j = 1 ; j <= n ; j++ )
  16.             {
  17.                 if ( region[i][j] > region[i][k] + region[k][j] )
  18.                 {
  19.                     region[i][j] = region[i][k] + region[k][j] ;
  20.                 }
  21.             }
  22.     }
  23. }
  24. int SearchPos( int m , int l )
  25. {
  26.     int i , j , k , min , sum , temp ;
  27.     min = MAX_INF ;
  28.     for ( k = 1 ; k <= m ; k++ )
  29.     {
  30.         sum = 0 ;
  31.         for ( i = 0 ; i < l ; i++ )
  32.         {
  33.             temp = MAX_INF ;
  34.             for ( j = 1 ; j <= m ; j++ )
  35.             {
  36.                 if ( place[gMember[i]][j] && temp > region[j][k] )
  37.                 {
  38.                     temp = region[j][k] ;
  39.                 }
  40.             }
  41.             sum += temp ;
  42.         }
  43.         if ( min > sum )
  44.         {
  45.             min = sum ;
  46.         }
  47.     }
  48.     return min ;
  49. }
  50. int main()
  51. {
  52.     int m , n , l , i , j , k , num , t ;
  53.     int pos[MAXN] , temp[MAXN][MAXN] ;
  54.     scanf("%d", &m) ;
  55.     scanf("%d", &n) ;
  56.     scanf("%d", &l) ;
  57.     for ( i = 0 ; i < l ; i++ )
  58.     {
  59.         scanf("%d", &gMember[i]) ;
  60.     }
  61.     memset(temp, 0, sizeof(temp)) ;
  62.     memset(place, 0, sizeof(place)) ;
  63.     
  64.     for ( i = 1 ; i <= m ; i++ )
  65.         for ( j = 1 ; j <= m ; j++ )
  66.         {
  67.             if ( i == j )
  68.                 region[i][j] = 0 ;
  69.             else 
  70.                 region[i][j] = MAX_INF ;
  71.         }
  72.     for ( i = 1 ; i <= m ; i++ )
  73.     {
  74.         scanf("%d", &num) ;
  75.         for ( j = 0 ; j < num ; j++ )
  76.         {
  77.             scanf("%d", &pos[j]) ;
  78.             place[pos[j]][i] = true ;
  79.         }
  80.         
  81.         for ( k = 0 ; k < num ; k++ )
  82.         {
  83.             j = pos[k] ;
  84.             if ( k == num - 1 )
  85.             {
  86.                 if ( temp[j][pos[0]] > 0 )
  87.                 {
  88.                     t = temp[j][pos[0]] ;
  89.                     region[i][t] = region[t][i] = 1 ;
  90.                 }
  91.                 else 
  92.                 {
  93.                     temp[j][pos[0]] = temp[pos[0]][j] = i ;
  94.                 }
  95.             }
  96.             else
  97.             {
  98.                 if ( temp[j][pos[k + 1]] > 0 )
  99.                 {
  100.                     t = temp[j][pos[k + 1]] ;
  101.                     region[i][t] = region[t][i] = 1 ;
  102.                 }
  103.                 else 
  104.                 {
  105.                     temp[j][pos[k + 1]] = temp[pos[k + 1]][j] = i ;
  106.                 }
  107.             }
  108.         }
  109.     }
  110.     Floyd( m ) ;
  111.     int ans = SearchPos( m , l ) ;
  112.     printf("%d/n", ans) ;
  113.     return 0 ;
  114. }
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值