[hdu5253] 最小生成树,Kruskal

题意:有n*m个单位的农田,给定每个单位农田地势高低,现在需要灌溉所有农田,如果把水引入相邻的农田里需要的管道长度为两者的高度差。求最少的管道长度花费。

思路:比较明显的最小生成树问题,相邻两点之间连一条边,边权为高度之差,求图的最小生成树即可。由于高度范围只有100,故可以直接类似hash表存了,无需排序。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
#include <iostream>
#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <vector>
#include <cmath>
#include <algorithm>
using  namespace  std;
 
vector<pair< int int > > C[101];
int  fa[1234567], a[1234567];
 
int  get_id( int  x,  int  y,  int  n) {  return  x * n + y; }
void  add( int  u,  int  v) {
     int  dif =  abs (a[u] - a[v]);
     C[dif].push_back(make_pair(u, v));
}
int  getfa( int  u) {  return  u == fa[u]? u : fa[u] = getfa(fa[u]); }
 
int  main() {
#ifndef ONLINE_JUDGE
     freopen ( "in.txt" "r" , stdin);
#endif // ONLINE_JUDGE
     int  T;
     cin >> T;
     for  ( int  cas = 1; cas <= T; cas ++) {
         int  n, m;
         cin >> n >> m;
         for  ( int  i = 0; i < 101; i ++) C[i].clear();
         for  ( int  i = 0; i < n * m; i ++) fa[i] = i;
         for  ( int  i = 0; i < n; i ++) {
             for  ( int  j = 0; j < m; j ++) {
                 scanf ( "%d" , a + get_id(i, j, m));
                 if  (i) add(get_id(i, j, m), get_id(i - 1, j, m));
                 if  (j) add(get_id(i, j, m), get_id(i, j - 1, m));
             }
         }
         int  cnt = 0, ans = 0;
         for  ( int  i = 0; i < 101; i ++) {
             int  sz = C[i].size();
             for  ( int  j = 0; j < sz; j ++) {
                 int  u = C[i][j].first, v = C[i][j].second;
                 int  fu = getfa(u), fv = getfa(v);
                 if  (fu != fv) {
                     ans += i;
                     cnt ++;
                     fa[fu] = fv;
                     if  (cnt == n * m - 1)  goto  END;
                 }
             }
         }
         END:
         printf ( "Case #%d:\n" , cas);
         cout << ans << endl;
     }
     return  0;
}

转载于:https://www.cnblogs.com/jklongint/p/4555954.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值