网络流的认识

网络流的认识

什么是流网络

网络(network)是指一个特殊的有向图  G = ( V , E ) G = (V,E) G=(V,E),其与一般有向图的不同之处在于有容量和源汇点,不考虑反向边。
其中,我们有以下变量来方便表示:

  • S S S:源点
  • T T T:汇点
  • c ( u , v ) c(u,v) c(u,v):表示从 u u u v v v 这条有向边的容量 c ( u , v ) c(u,v) c(u,v).
  • f ( u , v ) f(u,v) f(u,v):表示从 u u u v v v 这条有向边的流量 f ( u , v ) f(u,v) f(u,v).

没有网络连接到洛谷图床

图  1 \text{图 } 1  1
如上图,这就是一个流网络,其中 c ( 1 , 3 ) = 4 c(1,3)=4 c(1,3)=4 表示的就是 1 1 1 3 3 3 的容量为 4 4 4,源点 S S S 往汇点 T T T 进行流水操作,其中 S S S T T T 能容下水的量是无限量的。(这里的水只是打个比方,因为很像自来水工厂为我们供水)。

什么是可行流

可行流,通俗点讲,就是在每条变分配流水的多少,使能满足条件(这个在生活实际也能推出)。
条件显然为以下:

  • 流量限制: 0 ≤ f ( u , v ) ≤ c ( u , v ) 0\leq f(u,v)\leq c(u,v) 0f(u,v)c(u,v),你这条水管的流量如果大于容量,后果不堪设想。
  • 流量守恒:抽象点讲,也就是你当前的点为 u u u,入点为 p 1 , p 2 , … , p k 1 p_1,p_2,\dots,p_{k1} p1,p2,,pk1,出点分别为 q 1 , q 2 , … , q k 2 q_1,q_2,\dots,q_{k2} q1,q2,,qk2,那么显然:
    ∑ i = 1 i ≤ k 1 f ( p i , u ) = ∑ i = 1 i ≤ k 2 f ( u , q i ) \sum_{i=1}^{i\leq k1}f(p_i,u)=\sum_{i=1}^{i\leq k2}f(u,q_i) i=1ik1f(pi,u)=i=1ik2f(u,qi)
    形象点讲,也就是我当前流入到这个点的水的量最终都会流出到我可以到的点。

我们用 f f f 表示一个可行流,如下图:

有网络连接到洛谷图床
图  2 \text{图 } 2  2

其中在 1 1 1 3 3 3 这条边上, f ( 1 , 3 ) = 2 f(1,3) = 2 f(1,3)=2 c ( 1 , 3 ) = 4 c(1,3) = 4 c(1,3)=4,显然满足条件: 0 ≤ f ( 1 , 3 ) = 2 ≤ 4 = c ( 1 , 3 ) 0\leq f(1,3) = 2\leq 4 = c(1,3) 0f(1,3)=24=c(1,3).
你可以试试看,能否满足第二个条件?

什么是最大流

最大流(也称最大可行流)实际是在可行流中找一个方案,使得流入汇点 T T T 的水的量最大。我们用 ∣ f ∣ |f| f 表示 S S S T T T 点流量总和。根据定义显然有下种公式:
∣ f ∣ = ∑ ( u , v ) ∈ E f ( u , v ) − ∑ ( u , v ) ∈ E f ( v , u ) . |f| = \sum_{(u,v)\in E} f(u,v) - \sum_{(u,v)\in E} f(v,u). f=(u,v)Ef(u,v)(u,v)Ef(v,u).
在这里,右边的式子的第二个单项式 ∑ ( u , v ) ∈ E f ( v , u ) \sum_{(u,v)\in E} f(v,u) (u,v)Ef(v,u) 可以忽略不计,为了严谨,考虑了反向边的情况。

What is 残留网络

概念和构建

残留网络总是针对原图 G = ( V , E ) G =(V,E) G=(V,E) 中的某一个可行流而言,因此,可以将残留网络看成是可行流的一个函数,通常记为 G f G_f Gf.
G f = ( V f , E f ) G_f = (V_f,E_f) Gf=(Vf,Ef),其中 V f = V V_f =V Vf=V E f = E E_f = E Ef=E E E E 中所有的反向边。
残留网络中的容量记为 c ′ ( u , v ) c'(u,v) c(u,v),且 ( u , v ) ∈ E , ( v , u ) ∈ E (u,v) \in E,(v,u) \in E (u,v)E,(v,u)E,定义为:
c ′ ( u , v ) = { c ( u , v ) − f ( u , v ) ( u , v ) ∈ E f ( v , u ) ( u , v ) ∈ E c'(u,v) = \left\{\begin{matrix} c(u,v) - f(u,v)\quad (u,v)\in E\\ f(v,u) \quad (u,v)\in E \end{matrix}\right. c(u,v)={c(u,v)f(u,v)(u,v)Ef(v,u)(u,v)E

作用

可以通过 增广路径 的配合找到更大的流,使最后图中的最大流最大。

增广路径

定义

如果从源点 S S S 出发沿着残留网络中容量大于 0 0 0 的边走,可以走到汇点 T T T,那么将走过的边所组成的路径称为增广路径。
在这里我们发现:原网络可行流+残留网络可行流也是原网络的一个可行流
抽象点说(正式点说), f + f ′ f+f' f+f 属于 G G G 的一个可行流,且有:
∣ f + f ′ ∣ = ∣ f ∣ + ∣ f ′ ∣ |f + f'| = |f| + |f'| f+f=f+f

在网络中定点的一个划分,把所有顶点分成两个集合 S S S T T T,其中 S ∈ S , T ∈ T S\in S,T\in T SS,TT,而且有 S ∪ T = V , S ∩ T = ∅ S\cup T=V,S\cap T=\emptyset ST=V,ST=,记为 [ S , T ] [S,T] [S,T].

割的容量

指连接两个点集的边的容量总和,即 c ( S , T ) = ∑ u ∈ S ∑ v ∈ T c ( u , v ) c(S,T)=\sum_{u\in S}\sum_{v\in T}c(u,v) c(S,T)=uSvTc(u,v)

割的流量

指指连接两个点集的边的流量总和,
由上同理可得:
f ( S , T ) = ∑ u ∈ S ∑ v ∈ T ( c ( u , v ) − c ( v , u ) ) f(S,T) = \sum_{u\in S}\sum_{v\in T}(c(u,v)-c(v,u)) f(S,T)=uSvT(c(u,v)c(v,u))
有反向边时, c ( v , u ) c(v,u) c(v,u) 才有确值。
显然:
0 ≤ f ( S , T ) ≤ c ( S , T ) 0\leq f(S,T)\leq c(S,T) 0f(S,T)c(S,T)

最小割

G G G 中所有割组成的集合中,容量最小的元素。

最大流最小割定理

以下 3 3 3 个,知 1 1 1 2 2 2

  • 1 1 1 f f f 是最大流
  • 2 2 2 G f G_f Gf 不存在增广路
  • 3 3 3 ∃ [ S , T ] ∃[S,T] [S,T],满足 ∣ f ∣ = c ( S , T ) |f|=c(S,T) f=c(S,T) ∃ ∃ 表示存在一个)。

证明:

  • 证明 1 → 2 1\rightarrow 2 12
    反证即可,若存在增广路就会使得当前的 f f f 不是最大流,也就是 ∣ f ∣ + ∣ f ′ ∣ > ∣ f ∣ |f| + |f'|>|f| f+f>f,由条件又可知道: ∣ f ∣ |f| f 最大,所以说当 G f G_f Gf 不存在增广路时, f f f 为最大流。
  • 证明 2 → 3 2\rightarrow 3 23
    我们将对于 G f G_f Gf 中从 S S S 出发沿着容量大于 0 0 0 的边可以到达的点全部放入集合 S S S 中,然后令 T = V − S T = V - S T=VS,那么对于点 x ∈ S , y ∈ Y x\in S,y\in Y xS,yY,边 ( x , y ) (x,y) (x,y) 必有 f ( x , y ) = c ( x , y ) f(x,y) = c(x,y) f(x,y)=c(x,y)
  • 证明 3 → 1 3\rightarrow 1 31
    因为 ∣ f ∣ ≤ 最大流 ≤ c ( S , T ) |f|\leq \text{最大流}\leq c(S,T) f最大流c(S,T),而由 3 3 3 可知 ∣ f ∣ = c ( S , T ) |f|=c(S,T) f=c(S,T),故上式取等,即 f f f 是最大流。

求最大流

基于上述定理,我们可以不断寻找增广路,利用增广路更新残留网络,直到找不到增广路,即可求得最大流。

EK 算法

时间复杂度 O ( n m 2 ) O(nm^2) O(nm2)

改图
继续寻找
拿到原图
残留网络
找增广路
是否存在增广路到达 T
更新残留网络
找到最大流
输出
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 2 m ) O(n^2m) O(n2m)

#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;
}

完结。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值