题意:有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:
", cas); cout << ans << endl; } return 0;} |