网络流的认识
网络流的认识
什么是流网络
网络(network
)是指一个特殊的有向图 \(G = (V,E)\),其与一般有向图的不同之处在于有容量和源汇点,不考虑反向边。
其中,我们有以下变量来方便表示:
- \(S\):源点
- \(T\):汇点
- \(c(u,v)\):表示从 \(u\) 到 \(v\) 这条有向边的容量为 \(c(u,v)\).
- \(f(u,v)\):表示从 \(u\) 到 \(v\) 这条有向边的流量为 \(f(u,v)\).
如上图,这就是一个流网络,其中 \(c(1,3)=4\) 表示的就是 \(1\) 到 \(3\) 的容量为 \(4\),源点 \(S\) 往汇点 \(T\) 进行流水操作,其中 \(S\) 和 \(T\) 能容下水的量是无限量的。(这里的水只是打个比方,因为很像自来水工厂为我们供水)。
什么是可行流
可行流,通俗点讲,就是在每条变分配流水的多少,使能满足条件(这个在生活实际也能推出)。
条件显然为以下:
- 流量限制:\(0\leq f(u,v)\leq c(u,v)\),你这条水管的流量如果大于容量,后果不堪设想。
- 流量守恒:抽象点讲,也就是你当前的点为 \(u\),入点为 \(p_1,p_2,\dots,p_{k1}\),出点分别为 \(q_1,q_2,\dots,q_{k2}\),那么显然:
形象点讲,也就是我当前流入到这个点的水的量最终都会流出到我可以到的点。
我们用 \(f\) 表示一个可行流,如下图:
其中在 \(1\) 到 \(3\) 这条边上,\(f(1,3) = 2\),\(c(1,3) = 4\),显然满足条件:\(0\leq f(1,3) = 2\leq 4 = c(1,3)\).
你可以试试看,能否满足第二个条件?
什么是最大流
最大流(也称最大可行流)实际是在可行流中找一个方案,使得流入汇点 \(T\) 的水的量最大。我们用 \(|f|\) 表示 \(S\) 到 \(T\) 点流量总和。根据定义显然有下种公式:
在这里,右边的式子的第二个单项式 \(\sum_{(u,v)\in E} f(v,u)\) 可以忽略不计,为了严谨,考虑了反向边的情况。
What is 残留网络
概念和构建
残留网络总是针对原图 \(G =(V,E)\) 中的某一个可行流而言,因此,可以将残留网络看成是可行流的一个函数,通常记为 \(G_f\).
\(G_f = (V_f,E_f)\),其中 \(V_f =V\),\(E_f = E\) 和 \(E\) 中所有的反向边。
残留网络中的容量记为 \(c'(u,v)\),且 \((u,v) \in E,(v,u) \in E\),定义为:
作用
可以通过 增广路径
的配合找到更大的流,使最后图中的最大流最大。
增广路径
定义
如果从源点 \(S\) 出发沿着残留网络中容量大于 \(0\) 的边走,可以走到汇点 \(T\),那么将走过的边所组成的路径称为增广路径。
在这里我们发现:原网络可行流+残留网络可行流也是原网络的一个可行流
抽象点说(正式点说),\(f+f'\) 属于 \(G\) 的一个可行流,且有:
割
在网络中定点的一个划分,把所有顶点分成两个集合 \(S\) 和 \(T\),其中 \(S\in S,T\in T\),而且有 \(S\cup T=V,S\cap T=\emptyset\),记为 \([S,T]\).
割的容量
指连接两个点集的边的容量总和,即 \(c(S,T)=\sum_{u\in S}\sum_{v\in T}c(u,v)\)
割的流量
指指连接两个点集的边的流量总和,
由上同理可得:
有反向边时,\(c(v,u)\) 才有确值。
显然:
最小割
指 \(G\) 中所有割组成的集合中,容量最小的元素。
最大流最小割定理
以下 \(3\) 个,知 \(1\) 得 \(2\):
- \(1\)、\(f\) 是最大流
- \(2\)、\(G_f\) 不存在增广路
- \(3\)、\(∃[S,T]\),满足 \(|f|=c(S,T)\)(\(∃\)表示存在一个)。
证明:
- 证明 \(1\rightarrow 2\):
反证即可,若存在增广路就会使得当前的 \(f\) 不是最大流,也就是 \(|f| + |f'|>|f|\),由条件又可知道:\(|f|\) 最大,所以说当 \(G_f\) 不存在增广路时,\(f\) 为最大流。 - 证明 \(2\rightarrow 3\):
我们将对于 \(G_f\) 中从 \(S\) 出发沿着容量大于 \(0\) 的边可以到达的点全部放入集合 \(S\) 中,然后令 \(T = V - S\),那么对于点 \(x\in S,y\in Y\),边 \((x,y)\) 必有 \(f(x,y) = c(x,y)\)。 - 证明 \(3\rightarrow 1\):
因为 \(|f|\leq \text{最大流}\leq c(S,T)\),而由 \(3\) 可知 \(|f|=c(S,T)\),故上式取等,即 \(f\) 是最大流。
求最大流
基于上述定理,我们可以不断寻找增广路,利用增广路更新残留网络,直到找不到增广路,即可求得最大流。
EK 算法
时间复杂度 \(O(nm^2)\)
Code1
#include<iostream>
#include<cstring>
using namespace std;
const int INF=1e9;
const int N=1005, M=10010;
int n, m, S, T;
struct node{
int to, c, next;
}e[M<<1];
int h[N], tot;
// 残量网络建图,初始时正向的容量是 c, 反向容量是 0 。
void add(int u, int v, int c){
e[tot].to=v, e[tot].c=c, e[tot].next=h[u], h[u]=tot++;
e[tot].to=u, e[tot].c=0, e[tot].next=h[v], h[v]=tot++;
}
int lim[N], pre[N]; // lim[u] 表示 S 到点 u 路径容量的最小值, pre[u] 表示 u 的前驱边。
bool vis[N];
int q[N];
// bfs 找增广路。
bool bfs(){
memset(vis, false, sizeof vis);
int hh=0, tt=-1;
q[++tt]=S, vis[S]=true, lim[S]=INF;
while(tt>=hh){
int hd=q[hh++];
for(int i=h[hd]; ~i; i=e[i].next){
int go=e[i].to;
if(vis[go] || !e[i].c) continue;
vis[go]=true, q[++tt]=go;
lim[go]=min(lim[hd], e[i].c);
pre[go]=i;
if(go==T) return true;
}
}
return false;
}
int EK(){
int res=0;
while(bfs()){
res+=lim[T];
for(int i=T; i!=S; i=e[pre[i]^1].to){
e[pre[i]].c-=lim[T], e[pre[i]^1].c+=lim[T];
}
}
return res;
}
int main(){
memset(h, -1, sizeof h);
cin>>n>>m>>S>>T;
while(m--){
int u, v, c; cin>>u>>v>>c;
add(u, v, c);
}
cout<<EK()<<endl;
return 0;
}
Code2
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cmath>
#include <queue>
#include <cstring>
#define N 1007
#define M 10007
#define int long long
using namespace std;
const int INF = 1e9 + 7;
struct node{
int to,val,nxt;
}e[M << 1];
int n, m, S, T, tot;
int head[N],dis[N],pre[N];
bool vis[N];
queue<int> q;
void add(int a,int b,int c) {
e[tot].to = b, e[tot].val = c, e[tot].nxt = head[a], head[a] = tot++;
e[tot].to = a, e[tot].val = 0, e[tot].nxt = head[b], head[b] = tot++;
}
bool bfs() {
while(!q.empty()) q.pop();
memset(vis,false,sizeof vis);
q.push(S), vis[S] = true, dis[S] = INF;
while(!q.empty()) {
int t = q.front();
q.pop();
for (int i = head[t];i != -1;i = e[i].nxt) {
int v = e[i].to;
if (!vis[v] && e[i].val) {
vis[v] = true;
dis[v] = min(dis[t],e[i].val);
pre[v] = i;
if (v == T) return true;
q.push(v);
}
}
}
return false;
}
int EK(){
int r = 0;
while(bfs()) {
r += dis[T];
for (int i = T;i != S;i = e[pre[i] ^ 1].to)
e[pre[i]].val -= dis[T], e[pre[i] ^ 1].val += dis[T];
}
return r;
}
signed main(){
cin >>n >> m >> S>> T;
memset(head, -1,sizeof head);
for(;m--;) {
int a,b,c;
cin >> a >> b >> c;
add(a,b,c);
}
cout << EK();
return 0;
}
Dinic 算法
多路增广,时间复杂度 \(O(n^2m)\)
#include<bits/stdc++.h>
using namespace std;
#define int long long
#define gc() (st==ed&&(ed=(st=buf)+fread(buf,1,100000,stdin),st==ed)?EOF:*st++)
char buf[100001],*st=buf,*ed=buf;
void read(int &a){
a=0;char c=gc();
while(c>'9'||c<'0')c=gc();
while(c>='0'&&c<='9')a=a*10+c-48,c=gc();
}
const int INF=0x3f3f3f3f;
const int N=10010, M=1e5+5;
struct node{
int to, c, next;
}e[M<<1];
int h[N], tot;
void add(int u, int v, int cap){
e[tot].to=v, e[tot].c=cap, e[tot].next=h[u], h[u]=tot++;
e[tot].to=u, e[tot].c=0, e[tot].next=h[v], h[v]=tot++;
}
int n, m, S, T;
int d[N], q[N], cur[N];
bool bfs(){
memset(d, -1, sizeof d);
int tt=-1, hh=0;
q[++tt]=S, d[S]=0, cur[S]=h[S];
while(tt>=hh){
int hd=q[hh++];
for(int i=h[hd]; ~i; i=e[i].next){
int go=e[i].to;
if(d[go]==-1 && e[i].c){
d[go]=d[hd]+1;
cur[go]=h[go];
if(go==T) return true;
q[++tt]=go;
}
}
}
return false;
}
int find(int u, int limit){
if(u==T) return limit;
int flow=0;
for(int i=cur[u]; ~i && flow<limit; i=e[i].next){
cur[u]=i;
int go=e[i].to;
if(d[go]==d[u]+1 && e[i].c){
int t=find(go, min(e[i].c, limit-flow));
if(!t) d[go]=-1;
e[i].c-=t, e[i^1].c+=t, flow+=t;
}
}
return flow;
}
int dinic(){
int res=0, flow;
while(bfs()) while(flow=find(S, INF)) res+=flow;
return res;
}
signed main(){
memset(h, -1, sizeof h);
read(n), read(m), read(S), read(T);
while(m--){
int u, v, cap; read(u), read(v), read(cap);
add(u, v, cap);
}
cout<<dinic()<<endl;
return 0;
}
完结。