hoj 2715 (費用流 拆點)

http://acm.hit.edu.cn/hoj/problem/view?id=2715ios

將每一個格子 i 拆成兩個點 i’, i’’並加邊(i’, i’’, 1, -Vi), (i’, i’’, ∞, 0), (s, i’, ∞, 0); 控制只有一次能取到寶物。ui

對相鄰的四個格子 j, Hi > Hj 則加邊(i’’, j’, ∞, 0);spa

若格子 i 在邊界上則加邊(i’’, t, ∞, 0)。code

限制增廣次數小於等於 K 求最小費用流便可。blog

  1 #include <iostream>
  2 #include <cstdio>
  3 #include <cstring>
  4 #include <algorithm>
  5 #include <queue>
  6 #include <cmath>
  7 
  8 using namespace std;
  9 
 10 const int maxn = 505;
 11 const int maxm = 2000;
 12 const int inf = 0x3f3f3f3f;
 13 
 14 struct MCMF
 15 {
 16     struct Edge
 17     {
 18         int v, c, w, next;
 19     }p[maxm << 1];
 20     int e, head[maxn], dis[maxn], pre[maxn], cnt[maxn], sumFlow, n;
 21     bool vis[maxn];
 22     void init(int nt)
 23     {
 24         e = 0; n = nt;
 25         memset(head, -1, sizeof(head[0]) * (n + 2) );
 26     }
 27     void addEdge(int u, int v, int c, int w)
 28     {
 29         p[e].v = v; p[e].c = c; p[e].w = w; p[e].next = head[u]; head[u] = e++;
 30         swap(u, v);
 31         p[e].v = v; p[e].c = 0; p[e].w = -w; p[e].next = head[u]; head[u] = e++;
 32     }
 33     bool spfa(int S, int T)
 34     {
 35         queue <int> q;
 36         for (int i = 0; i <= n; ++i)
 37             vis[i] = cnt[i] = 0, pre[i] = -1, dis[i] = inf;
 38         vis[S] = 1, dis[S] = 0;
 39         q.push(S);
 40         while (!q.empty())
 41         {
 42             int u = q.front(); q.pop();
 43             vis[u] = 0;
 44             for (int i = head[u]; i + 1; i = p[i].next)
 45             {
 46                 int v = p[i].v;
 47                 if (p[i].c && dis[v] > dis[u] + p[i].w)
 48                 {
 49                     dis[v] = dis[u] + p[i].w;
 50                     pre[v] = i;
 51                     if (!vis[v])
 52                     {
 53                         q.push(v);
 54                         vis[v] = 1;
 55                         if (++cnt[v] > n) return 0;
 56                     }
 57                 }
 58             }
 59         }
 60         return dis[T] != inf;
 61     }
 62     int mcmf(int S, int T, int kt)
 63     {
 64         sumFlow = 0;
 65         int minFlow = 0, minCost = 0;
 66         while (spfa(S, T) && (kt--))
 67         {
 68             minFlow = inf + 1;
 69             for (int i = pre[T]; i + 1; i = pre[ p[i ^ 1].v ])
 70                 minFlow = min(minFlow, p[i].c);
 71             sumFlow += minFlow;
 72             for (int i = pre[T]; i + 1; i = pre[ p[i ^ 1].v ])
 73             {
 74                 p[i].c -= minFlow;
 75                 p[i ^ 1].c += minFlow;
 76             }
 77             minCost += dis[T] * minFlow;
 78         }
 79         return minCost;
 80     }
 81     void build(int nt, int kt)
 82     {
 83         int nnt = nt * nt;
 84         init(nnt * 2 + 1);
 85         int val[53][53], height[53][53];
 86         memset(val, 0x3f, sizeof(val));
 87         memset(height, 0x3f, sizeof(height));
 88         for (int i = 1; i <= nt; ++i)
 89             for (int j = 1; j <= nt; ++j)
 90                 scanf("%d", &val[i][j]);
 91         for (int i = 1; i <= nt; ++i)
 92             for (int j = 1; j <= nt; ++j)
 93                 scanf("%d", &height[i][j]);
 94         for (int i = 1; i <= nt; ++i)
 95             for (int j = 1; j <= nt; ++j)
 96             {
 97                 int pos = nt * (i - 1) + j;
 98                 addEdge(0, pos, inf, 0);
 99                 addEdge(pos, pos + nnt, 1,-val[i][j]);
100                 addEdge(pos, pos + nnt, inf,0);
101                 if (i == 1 || i == nt || j == 1 || j == nt)
102                     addEdge(pos + nnt, n, inf, 0);
103                 if (height[i][j] > height[i][j - 1])
104                     addEdge(pos + nnt, pos - 1, inf, 0);
105                 if (height[i][j] > height[i][j + 1])
106                     addEdge(pos + nnt, pos + 1, inf, 0);
107                 if (height[i][j] > height[i - 1][j])
108                     addEdge(pos + nnt, pos - nt, inf, 0);
109                 if (height[i][j] > height[i + 1][j])
110                     addEdge(pos + nnt, pos + nt, inf, 0);
111             }
112     }
113     void solve(int nt, int kt)
114     {
115         build(nt, kt);
116         printf("%d\n", - mcmf(0, n, kt));
117     }
118 }my;
119 int main()
120 {
121     int tcase, n, k;
122     scanf("%d", &tcase);
123     while (tcase--)
124     {
125         scanf("%d%d", &n, &k);
126         my.solve(n, k);
127     }
128     return 0;
129 }
相關文章
相關標籤/搜索