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 58 59 60 61 62 63 64 65 66
| template <const int MAXV, const int MAXE> struct MCMF { const int INF = 0x3f3f3f3f3f3f3f3f; struct Edge { int v, flow, cost; } edge[MAXE * 2]; int tot = 1, head[MAXV], nxt[MAXE]; int flow, cost, cur[MAXV], dis[MAXV]; bool vis[MAXV]; void addedge(int u, int v, int flow, int cost) { edge[++tot] = {v, flow, cost}; nxt[tot] = head[u], head[u] = tot; edge[++tot] = {u, 0, -cost}; nxt[tot] = head[v], head[v] = tot; } bool spfa(int s, int t) { fill(vis, vis + MAXV, 0); fill(dis, dis + MAXV, INF); queue<int> que; que.push(s); dis[s] = 0; vis[s] = 1; while (!que.empty()) { int u = que.front(); que.pop(); vis[u] = 0; for (int i = head[u]; i; i = nxt[i]) { int v = edge[i].v; if (edge[i].flow && dis[v] > dis[u] + edge[i].cost) { dis[v] = dis[u] + edge[i].cost; if (!vis[v]) { que.push(v); vis[v] = 1; } } } } return dis[t] != INF; } int augment(int u, int t, int mx) { if (u == t || mx == 0) return mx; vis[u] = 1; int ret = 0; for (int &i = cur[u]; i; i = nxt[i]) { int v = edge[i].v; if (vis[v] || dis[v] != dis[u] + edge[i].cost) continue; int tmp = augment(v, t, min(mx, edge[i].flow)); cost += tmp * edge[i].cost; mx -= tmp, ret += tmp; edge[i].flow -= tmp, edge[i ^ 1].flow += tmp; if (mx == 0) break; } vis[u] = 0; return ret; } pair<int, int> mcmf(int s, int t) { while (spfa(s, t)) { copy(head, head + MAXV, cur); flow += augment(s, t, INF); } return make_pair(flow, cost); } };
|
Gitalking ...