在农夫约翰的农场上,每逢下雨,贝茜最喜欢的三叶草地就积聚了一潭水。这意味着草地被水淹没了,并且小草要继续生长还要花相当长一段时间。因此,农夫约翰修建了一套排水系统来使贝茜的草地免除被大水淹没的烦恼(不用担心,雨水会流向附近的一条小溪)。作为一名一流的技师,农夫约翰已经在每条排水沟的一端安上了控制器,这样他可以控制流入排水沟的水流量。 农夫约翰知道每一条排水沟每分钟可以流过的水量,和排水系统的准确布局(起点为水潭而终点为小溪的一张网)。需要注意的是,有些时候从一处到另一处不只有一条排水沟。 根据这些信息,计算从水潭排水到小溪的最大流量。对于给出的每条排水沟,雨水只能沿着一个方向流动,注意可能会出现雨水环形流动的情形。
第1行: 两个用空格分开的整数N (0 <= N <= 200) 和 M (2 <= M <= 200)。N是农夫约翰已经挖好的排水沟的数量,M是排水沟交叉点的数量。交点1是水潭,交点M是小溪。 第二行到第N+1行: 每行有三个整数,Si, Ei, 和 Ci。Si 和 Ei (1 <= Si, Ei <= M) 指明排水沟两端的交点,雨水从Si 流向Ei。Ci (0 <= Ci <= 10,000,000)是这条排水沟的最大容量。
输出一个整数,即排水的最大流量。
Sap网络流模板题,直接做就好,注意可能有重边。
#include<iostream> #include<cstdio> #include<cstdlib> #include<cstring> #include<algorithm> using namespace std; const int Max = 205; const int INF = 0x3f3f3f3f; int N, M, Flow, Src, Tar, All; int map[Max][Max]; int Vd[Max], D[Max]; void getint(int & num){ char c; int flg = 1; num = 0; while((c = getchar()) < '0' || c > '9') if(c == '-') flg = -1; while(c >= '0' && c <= '9'){ num = num * 10 + c - 48; c = getchar();} num *= flg; } int Aug(int i, int augco){ if(i == Tar) return augco; int augc = augco, delta, mind = All - 1; for(int j = Src; j <= All; ++ j) if(map[i][j] > 0){ if(D[i] == D[j] + 1){ delta= Aug(j, min(augc, map[i][j])); map[i][j] -= delta; map[j][i] += delta; augc -= delta; if(! augc) break; if(D[Src] >= All) return augco - augc; } if(mind > D[j]) mind = D[j]; } if(augc == augco){ -- Vd[D[i]]; if(! Vd[D[i]]) D[Src] = All; D[i] = mind + 1; ++ Vd[D[i]]; } return augco - augc; } int Sap(){ Vd[0] = All; Flow = 0; while(D[Src] < All) Flow += Aug(Src, INF); return Flow; } int main(){ getint(M), getint(N); Src = 1, Tar = N, All = N; int a, b, c; while(M --){ getint(a), getint(b), getint(c); map[a][b] += c; } printf("%d\n", Sap()); return 0; }