// UVa 10779 - Collector's Problem
#include <iostream>
#include <stdio.h>
#include <vector>
#include <string.h>
#include <cmath>
#include <queue>
using namespace std;
#define MaxN 10
#define MaxM 25
int person[MaxN][MaxM + 1];
typedef long long LL;
struct Edge {
int from, to, cap, flow, index;
Edge(int from, int to, int cap, int flow, int index) :
from(from), to(to), cap(cap), flow(flow), index(index) {
}
};
struct PushRelabel {
int N;
vector<vector<Edge> > G;
vector<LL> excess;
vector<int> dist, active, count;
queue<int> Q;
PushRelabel(int N) :
N(N), G(N), excess(N), dist(N), active(N), count(2 * N) {
}
void AddEdge(int from, int to, int cap) {
G[from].push_back(Edge(from, to, cap, 0, G[to].size()));
if (from == to)
G[from].back().index++;
G[to].push_back(Edge(to, from, 0, 0, G[from].size() - 1));
}
void Enqueue(int v) {
if (!active[v] && excess[v] > 0) {
active[v] = true;
Q.push(v);
}
}
void Push(Edge &e) {
int amt = int(min(excess[e.from], LL(e.cap - e.flow)));
if (dist[e.from] <= dist[e.to] || amt == 0)
return;
e.flow += amt;
G[e.to][e.index].flow -= amt;
excess[e.to] += amt;
excess[e.from] -= amt;
Enqueue(e.to);
}
void Gap(int k) {
for (int v = 0; v < N; v++) {
if (dist[v] < k)
continue;
count[dist[v]]--;
dist[v] = max(dist[v], N + 1);
count[dist[v]]++;
Enqueue(v);
}
}
void Relabel(int v) {
count[dist[v]]--;
dist[v] = 2 * N;
for (int i = 0; i < G[v].size(); i++)
if (G[v][i].cap - G[v][i].flow > 0)
dist[v] = min(dist[v], dist[G[v][i].to] + 1);
count[dist[v]]++;
Enqueue(v);
}
void Discharge(int v) {
for (int i = 0; excess[v] > 0 && i < G[v].size(); i++)
Push(G[v][i]);
if (excess[v] > 0) {
if (count[dist[v]] == 1)
Gap(dist[v]);
else
Relabel(v);
}
}
LL GetMaxFlow(int s, int t) {
count[0] = N - 1;
count[N] = 1;
dist[s] = N;
active[s] = active[t] = true;
for (int i = 0; i < G[s].size(); i++) {
excess[s] += G[s][i].cap;
Push(G[s][i]);
}
while (!Q.empty()) {
int v = Q.front();
Q.pop();
active[v] = false;
Discharge(v);
}
LL totflow = 0;
for (int i = 0; i < G[s].size(); i++)
totflow += G[s][i].flow;
return totflow;
}
};
PushRelabel pushRelabel(0);
int main() {
int cases;
scanf("%d", &cases);
for (int cas = 1; cas <= cases; cas++) {
memset(person, 0, sizeof(person));
// read
int n, m;
scanf("%d%d", &n, &m);
for (int i = 0; i < n; i++) {
int j;
for (scanf("%d", &j); j; j--) {
int k;
scanf("%d", &k);
person[i][k]++;
}
}
// build graph
pushRelabel = PushRelabel(n + m + 1);
for (int j = 1; j <= m; j++)
pushRelabel.AddEdge(0, n - 1 + j, person[0][j]);
for (int i = 1; i < n; i++)
for (int j = 1; j <= m; j++)
if (person[i][j] > 0)
pushRelabel.AddEdge(i, n - 1 + j, person[i][j] - 1);
else
pushRelabel.AddEdge(n - 1 + j, i, 1);
for (int j = 1; j <= m; j++)
pushRelabel.AddEdge(n - 1 + j, n + m, 1);
// solve
int sol = pushRelabel.GetMaxFlow(0, n + m);
printf("Case #%d: %d\n", cas, sol);
}
return 0;
}
Thursday, May 14, 2015
UVa 10779 - Collector's Problem
Subscribe to:
Post Comments (Atom)
No comments:
Post a Comment