SGU192 有上下界的网络流

题目传送门

无源汇有上下界可行流问题。

iniin_i表示至少流到i的流量,outiout_i表示至少从i流出的流量。

对于每一条从u到v,流量上界为r,下界为l的边,容量存r-l,然后inv=inv+lin_v = in_v + loutu=outu+lout_u = out_u + l

建超级源点s,超级汇点t,对于每个满足ini>outiin_i > out_i的i,从s向i连容量为inioutiin_i-out_i的边,对于每个满足outi>iniout_i>in_i的i,从i向t连容量为outiiniout_i-in_i的边,

接着从s向t跑最大流。

如果流量 从s连出来的边容量总和,则不行。

否则:每条边的流量为流量下界+新图中该边的流量。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
/*
Author: CNYALI_LK
LANG: C++
PROG: sgu194.cpp
*/
#include<bits/stdc++.h>
#define up(a,b,c) for (int a(b),end##a(c);a<=end##a;++a)
#define down(a,c,b) for (int a(b),end##a(c);a>=end##a;--a)
#define uup(a,c,b) for (int a(b),end##a(c);a<=end##a;++a)
#define udown(a,b,c) for (int a(b),end##a(c);a>=end##a;--a)
#define endl '\n'
#define debug(...) fprintf(stderr,__VA_ARGS__)
using namespace std;
const double eps=1e-8;
const double PI=acos(-1.0);
typedef long long ll;
struct inout{
//io优化省略
};
inout io;
template<class T>void chkmin(T &a,T b){a=a<b?a:b;}
template<class T>void chkmax(T &a,T b){a=a>b?a:b;}
int to[53333],lst[53333],w[53333],beg[233],e,s,t;
#define rev(x) (((x-1)^1)+1)
int out[233],in[233],sf[53333];
void add(int u,int v,int wea){
to[++e]=v;
lst[e]=beg[u];
w[e]=wea;
beg[u]=e;
}
int dis[233],q[233],cur[233],n;
int bfs(){
up(i,1,n)dis[i]=0,cur[i]=beg[i];
dis[s]=1;
int l,r,u,v;
l=r=1;
q[l]=s;
while(l<=r){
u=q[l];
for(int i=beg[u];i;i=lst[i])if(!dis[to[i]]&&w[i]){
dis[to[i]]=dis[u]+1;
q[++r]=to[i];
}
++l;
}
return dis[t];
}
int dfs(int x,int flw){
int f,i;
if(x==t){
return flw;
}
for(;cur[x];cur[x]=lst[cur[x]]){
i=cur[x];
if(w[i]&&dis[to[i]]==dis[x]+1){
if(f=dfs(to[i],min(flw,w[i]))){
w[i]-=f;
w[rev(i)]+=f;
return f;
}
}
}
return 0;
}
int main(){
#ifdef cnyali_lk
freopen("in.txt","r",stdin);
freopen("out.txt","w",stdout);
#endif
io.init();
int m,u,v,ww,l,r;
io>>n>>m;
s=n+1;
t=n+2;
up(i,1,m){
io>>u>>v>>l>>r;
out[u]+=l;
in[v]+=l;
ww=r-l;
add(u,v,ww);
add(v,u,0);
sf[i]=l;
}
int tot=0;
up(i,1,n)if(in[i]>out[i]){add(n+1,i,in[i]-out[i]);add(i,n+1,0);tot+=in[i]-out[i];}else if(in[i]<out[i]){add(i,n+2,out[i]-in[i]);add(n+2,i,0);}
n+=2;
int f,sum=0;
while(bfs()){
while(f=dfs(s,0x3f3f3f3f)){sum+=f;}
}
if(sum!=tot)io<<"NO\n";
else{
io<<"YES\n";
up(i,1,m){
io<<w[i*2]+sf[i]<<endl;
}
}
io.out();
return 0;
}
文章目录
|

博客使用Disqus作为评论系统