最大流和割的关系:
定理一:如果f是网络中的一个流,CUT(S,T)是任意一个割,那么f的值等于正向割边的流量与负向割边的流量之差。
证明:
设X和Y是网络中的两个顶点集合,用f(X,Y)表示从X中的一个顶点指向Y的一个顶点的所有弧(弧尾在X中,弧头在Y中:XY)的流量和。
只需证明:f=f(S,T)-f(T,S) 即可。下列结论成立:
如果X∩Y= ,那么:
f(X,(Y1∪Y2))=f(X,Y1)+f(X,Y2)
f((X1∪X2),Y)=f(X1,Y)+f(X2,Y) 成立。
根据网络流的特点:
如果V既不是源点也不是汇点,那么:
f({V},S∪T)-f(S∪T,{V})=0,可知任何一个点,流入的与流出的量相等。
如果V是源,那么:
f({V},S∪T)-f(S∪T,{V})=f
对于S中的所有点V都有上述关系式,相加得到:
f(S,S∪T)-f(S∪T,S)=f
又因为:
f(S,S∪T)-f (S∪T,S)
= (f(S,S)+f (S,T))-(f(S,S) +f (T,S))
= f(S,T)- f(T,S)
所以:f= f(S,T)- f(T,S) 定理得证。
f= f(S,T)- f(T,S)<=f(S,T)<=割CUT(S,T)的容量。
推论1:如果f是网络中的一个流,CUT(S,T)是一个割,那么f的值不超过割CUT(S,T)的容量。
推论2:网络中的最大流不超过任何割的容量
定理2:在任何网络中,如果f是一个流,CUT(S,T)是一个割,且f的值等于割CUT(S,T)的容量,那么f是一个最大流,CUT(S,T)是一个最小割(容量最小的割)。
证明:
令割CUT(S,T)的容量为C,所以流f的流量也为C。
假设另外的任意流f1,流量为c1,根据流量不超过割的容量,则c1<=c,所以f是最大流。
假设另外的任意割CUT(S1,T1),容量为c1,根据流量不超过割的容量,所以有c1>=c,故,CUT(S,T)是最小割。
定理3:最大流最小割定量:在任何的网络中,最大流的值等于最小割的容量。
结论1:最大流时,最小割cut(S,T)中,正向割边的流量=容量,逆向割边的流量为0。否则还可以增广。
结论2:在最小割中cut(S,T)中:
① 源点s∈S。
② 如果i∈S,结点j满足:有弧,并且c[I,j]>f[I,j]或者有弧
Ford-Fulkerson算法的实现:
#include
#include
using namespace std;
#define MAX_N 1001
#define INF 0xffffff
#define min(a, b) (a < b ? a : b)
struct edge {
int to, cap;
int rev;
edge(int t, int c, int r) {
to = t;
cap = c;
rev = r;
}
};
vector G[MAX_N];
bool visited[MAX_N];
int n, m;
// 每对顶点添加正反边
void addEdge(int from, int to, int cap) {
G[from].push_back(edge(to, cap, G[to].size()));
G[to].push_back(edge(from, 0, G[from].size() - 1));
}
int dfs(int v, int t, int f) {
int i;
if (v == t) {
return f;
}
visited[v] = true;
for (i = 0; i < G[v].size(); i++) {
edge &e = G[v][i];
if (!visited[e.to] && e.cap > 0) {
int d = dfs(e.to, t, min(f, e.cap));
if (d > 0) {
e.cap -= d;
G[e.to][e.rev].cap += d;
return d;
}
}
}
return 0;
}
int maxFlow(int s, int t) {
int flow = 0;
while (true) {
memset(visited, 0, sizeof(visited));
int f = dfs(s, t, INF);
if (f == 0) {
return flow;
}
flow += f;
}
}
int main() {
int i, j;
int from, to, cap;
while (scanf("%d%d", &n, &m) != EOF && n || m) {
for (i = 0; i < n; i++) {
G[i].clear();
}
for (i = 0; i < m; i++) {
scanf("%d%d%d", &from, &to, &cap);
addEdge(from, to, cap);
}
printf("%d\n", maxFlow(0, n - 1));
}
return 0;
}
5 7
0 1 10
0 2 2
1 2 6
1 3 6
2 4 5
3 2 3
3 4 8