标签:
4 6 1 0 0 1 0 0 0 1 1 0 0 0 2 0 0 0 0 0 0 2 0 1 1 0
Case 1: 4
题意:n*m的图中1表示羊,2表示狼,要吧狼还有羊隔离,需要建篱笆,每一段篱笆需要1的花费,问需要的最小花费是多少
没学最小割之前真不懂啊,设立超级源点还有超级汇点,超级源点连所有的羊,超级汇点连所有的狼,边权均为INF表示该点不能删,狼还有羊可以向四周建边,边权为1,也是花费为1,0可以当做是中间节点,通过0浪可以到达羊,题目就变成要阻断所有的狼需要的花费
#include <cstdio> #include <cstring> #include <queue> #include <stack> #include <vector> #include <algorithm> #define MAXN 40000+10 #define MAXM 1000000+10 #define INF 0x3f3f3f3f using namespace std; struct Edge { int from, to, cap, flow, next; }; Edge edge[MAXM]; int head[MAXN], edgenum; int dist[MAXN]; int cur[MAXN]; bool vis[MAXN]; int N, M; int source, sink; void init() { edgenum = 0; memset(head, -1, sizeof(head)); } int point(int x, int y) { return (x-1) * M + y; } void addEdge(int u, int v, int w) { Edge E1 = {u, v, w, 0, head[u]}; edge[edgenum] = E1; head[u] = edgenum++; Edge E2 = {v, u, 0, 0, head[v]}; edge[edgenum] = E2; head[v] = edgenum++; } bool judge(int x, int y) { return x >= 1 && x <= N && y >= 1 && y <= M; } void getMap() { int a; source = 0, sink = N * M + 1; int move[4][2] = {0,1, 0,-1, 1,0, -1,0}; for(int i = 1; i <= N; i++) { for(int j = 1; j <= M; j++) { scanf("%d", &a); for(int p = 0; p < 4; p++) { int x = i + move[p][0]; int y = j + move[p][1]; if(judge(x, y)) addEdge(point(i, j), point(x, y), 1); } if(a == 1)//? addEdge(source, point(i, j), INF);//? ???? else if(a == 2)//? addEdge(point(i, j), sink, INF);//????? } } } bool BFS(int s, int t) { queue<int> Q; memset(dist, -1, sizeof(dist)); memset(vis, false, sizeof(vis)); dist[s] = 0; vis[s] = true; Q.push(s); while(!Q.empty()) { int u = Q.front(); Q.pop(); for(int i = head[u]; i != -1; i = edge[i].next) { Edge E = edge[i]; if(!vis[E.to] && E.cap > E.flow) { dist[E.to] = dist[u] + 1; if(E.to == t) return true; vis[E.to] = true; Q.push(E.to); } } } return false; } int DFS(int x, int a, int t) { if(x == t || a == 0) return a; int flow = 0, f; for(int &i = cur[x]; i != -1; i = edge[i].next) { Edge &E = edge[i]; if(dist[E.to] == dist[x] + 1 && (f = DFS(E.to, min(a, E.cap-E.flow), t)) > 0) { edge[i].flow += f; edge[i^1].flow -= f; flow += f; a -= f; if(a == 0) break; } } return flow; } int Maxflow(int s, int t) { int flow = 0; while(BFS(s, t)) { memcpy(cur, head, sizeof(head)); flow += DFS(s, INF, t); } return flow; } int main() { int k = 1; while(scanf("%d%d", &N, &M) != EOF) { init(); getMap(); printf("Case %d:\n%d\n", k++, Maxflow(source, sink)); } return 0; }
hdoj--3046--Pleasant sheep and big big wolf(最小割经典)
标签:
原文地址:http://blog.csdn.net/qq_29963431/article/details/51356051