先导知识

网络最大流

题目链接

https://vjudge.net/problem/ZOJ-2314

题目大意

多组数据,第一行为数据组数 \(T\)。

对于每一组数据,第一行为 \(n,m\) 表示 \(n\) 个结点,\(m\) 条有向边。

接下来 \(m\) 行,每一行有四个正整数 \(i,j,l_{ij},c_{ij}\) ,表示有一条从\(i\)到\(j\)的有向边,要求正整数流量 \(f_{ij} \in [l_{ij},c_{ij}]\) 。

题目还提到,不存在自环;以及如果存在 \(i\) 到 \(j\) 的边,则不存在 \(j\) 到 \(i\) 的反向边。

如果有解输出 \(YES\) ,并按顺序输出每一条边的可行流 \(f_k\) ,否则输出 \(NO\) 。

题目解析

题目为无源汇有上下界的可行流模板。

要点在于:

将每一条边用其 下界流 填满,算出每条边的剩余容量 \(cap_{ij} = c_{ij}-l_{ij}\) 以及每个结点的流量盈余 (\(R[x] = out_{\sum {l_{x*}}} - in_{\sum {l_{*x}}}\))。

新设置两个点——超级源点 \(s\) 和超级汇点 \(t\)。

对于结点 \(x\),如果盈余为正,连接 \(s\rightarrow x\) ;如果盈余为负,连接 \(x\rightarrow t\) ,容量为\(|R[x]|\)。(这是本题的关键所在,为什么可以这样等效?)

对于新图,跑最大流即可。(本题解用的是\(ISAP\),用\(Dinic\)当然也可以啦。)

如果发现源点\(s\)出发的边不是满流(\(\exists i, cap_{si} > flow_{si}\)),则证明原图可行流不存在。

否则可以得到答案(\(f_{ij} = l_{ij} + flow_{ij}\))。

例如:

Sample Input

4 6

1 2 1 2

2 3 1 2

3 4 1 2

4 1 1 2

1 3 1 2

4 2 1 2

可以化作下图。

算出边的剩余容量 \(cap_{ij}\) (以黑色数字标注在边上)和结点的流量盈余 \(R[x]\) (标注在点上,红色为正,蓝色为负)。

然后连结超级源点 \(s\) 和超级汇点 \(t\) 。

通过这种方法可以转化成一般的最大流问题。

这样做为什么正确呢?

其实可以形象化地理解,设置源点 \(s\) 和汇点 \(t\) 是两个虚拟结点。

对于结点 \(x\),为了保证除了源汇两点以外的点流量守恒,即 \(out_{\sum {l_{x*}}} = in_{\sum {l_{*x}}}\) 。

如果 \(R[x]>0\) ,盈余为正,可以看作这部分的盈余是从源点 \(s\) 流入的,所以连接 \(s\rightarrow x\) ;

如果 \(R[x]<0\) ,发现“入不敷出”(盈余为负),可以看作这部分的亏损是流入汇点 \(t\) 导致的,所以连接 \(x\rightarrow t\) 。

最后跑完最大流,则检查这些虚拟的边是否“满流”。

如果不能满流,则说明盈余来自源点/亏损流向汇点的假设不能成立,存在无法满足流量守恒的结点,该网络图自然不存在可行流。

如果能够满流,则说明这样假设是能够成立的,网络中的流量就是可行流。

由于虚拟的源汇依然有出流=入流,因此只要检查从源点 \(s\) 流出的边(或者流出汇点 \(t\) 的边)是否满流即可。

参考代码

#include <bits/stdc++.h>
#define ll long long
using namespace std;
const int N = 255;
const int INF = 2147483647;
struct Edge{
int from, to, cap, flow, min;
};
int n, m, s, t, gap[N], cur[N], dep[N], R[N];
vector <Edge> e;
vector <int> G[N]; void init()
{
memset(gap, 0, sizeof gap);
memset(cur, 0, sizeof cur);
memset(dep, 0, sizeof dep);
++gap[dep[t] = 1];
queue <int> Q;
Q.push(t);
while (!Q.empty()) {
int x=Q.front(); Q.pop();
for (int i = 0; i < G[x].size(); i++)
{
int v = e[G[x][i]].to;
if (!dep[v])
{
++gap[dep[v] = dep[x]+1];
Q.push(v);
}
}
}
}
int augment(int x, int a)
{
if (x == t || !a) return a;
int flow = 0;
for (int &i=cur[x]; i < G[x].size(); i++)
{
Edge& b = e[G[x][i]];
if (dep[x] == dep[b.to] + 1 && b.cap > b.flow) {
int tmp = augment(b.to, min(a, b.cap - b.flow));
flow += tmp;
a -= tmp;
b.flow += tmp;
e[G[x][i]^1].flow -= tmp;
if (!a) return flow;
}
}
if (!(--gap[dep[x]])) dep[s] = n+1;
++gap[++dep[x]], cur[x] = 0;
return flow;
}
ll maxFlow()
{
init();
ll ans = 0;
while (dep[s] <= n) ans += augment(s, INF);
for (int i = 0; i < G[s].size(); ++i) {
if (e[G[s][i]].cap > e[G[s][i]].flow) return -1;
}
return ans;
}
void addEdge(int u, int v, int l, int c, int i)
{
e.push_back((Edge){u, v, c-l, 0, l});
e.push_back((Edge){v, u, 0, 0, 0});
G[u].push_back(i);
G[v].push_back(i^1);
R[u] -= l;
R[v] += l;
}
void Clear()
{
e.clear();
for (int i = 0; i <= n; ++i) G[i].clear();
memset(R, 0, sizeof R);
n = m = s = t = 0;
}
int main()
{
int T;
scanf("%d", &T);
for (int Kase=0; Kase<T; ++Kase)
{
if (Kase) {Clear(); putchar('\n');}
scanf("%d%d", &n, &m);
for (int i = 0; i < m; ++i) {
int u, v, l ,c;
scanf("%d%d%d%d", &u, &v, &l, &c);
addEdge(u, v, l, c, i << 1);
}
s = ++n, t = ++n;
int j = m-1;
for (int i = 1; i <= n; ++i) {
if (R[i] > 0) addEdge(s, i, 0, R[i], (++j) << 1);
else if (R[i] < 0) addEdge(i, t, 0, -R[i], (++j) << 1);
}
ll a = maxFlow(); // printf("maxflow = %lld\n", a);
// for (int i = 0; i <= j; ++i) {
// printf("%d -> %d (%d/%d)\n", e[i << 1].from, e[i << 1].to, e[i << 1].flow, e[i << 1].cap);
// } //这一部分查看可以看新图的流量情况 if (a == -1) {printf("NO\n"); continue;}
printf("YES\n");
for (int i = 0; i < m; ++i)
printf("%d\n", e[i << 1].flow+e[i << 1].min);
}
return 0;
}

感谢支持!

【模板】无源汇有上下界可行流(网络流)/ZOJ2314的更多相关文章

  1. [loj#115] 无源汇有上下界可行流 网络流

    #115. 无源汇有上下界可行流 内存限制:256 MiB时间限制:1000 ms标准输入输出 题目类型:传统评测方式:Special Judge 上传者: 匿名 提交提交记录统计讨论测试数据   题 ...

  2. LOJ.115.[模板]无源汇有上下界可行流(Dinic)

    题目链接 参考:http://blog.csdn.net/clove_unique/article/details/54884437 http://blog.csdn.net/wu_tongtong/ ...

  3. 2018.08.20 loj#115. 无源汇有上下界可行流(模板)

    传送门 又get到一个新技能,好兴奋的说啊. 一道无源汇有上下界可行流的模板题. 其实这东西也不难,就是将下界变形而已. 准确来说,就是对于每个点,我们算出会从它那里强制流入与流出的流量,然后与超级源 ...

  4. 【LOJ115】无源汇有上下界可行流(模板题)

    点此看题面 大致题意: 给你每条边的流量上下界,让你判断是否存在可行流.若有,则还需输出一个合法方案. 大致思路 首先,每条边既然有一个流量下界\(lower\),我们就强制它初始流量为\(lower ...

  5. loj#115. 无源汇有上下界可行流

    \(\color{#0066ff}{ 题目描述 }\) 这是一道模板题. \(n\) 个点,\(m\) 条边,每条边 \(e\) 有一个流量下界 \(\text{lower}(e)\) 和流量上界 \ ...

  6. Zoj 2314 Reactor Cooling(无源汇有上下界可行流)

    http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=1314 题意:    给n个点,及m根pipe,每根pipe用来流躺液体的,单向 ...

  7. LOJ [#115. 无源汇有上下界可行流](https://loj.ac/problem/115)

    #115. 无源汇有上下界可行流 先扔个板子,上下界的东西一点点搞,写在奇怪的合集里面 Code: #include <cstdio> #include <cstring> # ...

  8. LibreOJ #115. 无源汇有上下界可行流

    二次联通门 : LibreOJ #115. 无源汇有上下界可行流 /* LibreOJ #115. 无源汇有上下界可行流 板子题 我也就会写写板子题了.. */ #include <cstdio ...

  9. 无源汇有上下界可行流(ZQU 1590)

    无源汇有上下界可行流(也就是循环流) 模型:一个网络,求出一个流,使得每条边的流量必须>=Li且<=Hi, 每个点必须满足总流入量=总流出量(流量守恒)(这个流的特点是循环往复,无始无终) ...

随机推荐

  1. [BZOI2014]大融合——————线段树进阶

    竟然改了不到一小时就改出来了, 可喜可贺 Description Solution 一开始想的是边两侧简单路径之和的乘积,之后发现这是个树形结构,简单路径数就是节点数. 之后的难点就变成了如何求线段树 ...

  2. 必备的60个常用的Linux命令

    Linux必学的60个命令Linux提供了大量的命令,利用它可以有效地完成大量的工 作,如磁盘操作.文件存取.目录操作.进程管理.文件权限设定等.所以,在Linux系统上工作离不开使用系统提供的命令. ...

  3. SpringCloud微服务实战——搭建企业级开发框架(十二):OpenFeign+Ribbon实现负载均衡

      Ribbon是Netflix下的负载均衡项目,它主要实现中间层应用程序的负载均衡.为Ribbon配置服务提供者地址列表后,Ribbon就会基于某种负载均衡算法,自动帮助服务调用者去请求.Ribbo ...

  4. 【数据结构&算法】02-复杂度分析之执行效率和资源消耗

    目录 前言 复杂度 分析方法 大 O 复杂度表示法 例子-评估累加和的各种算法执行效率 算法 1(for 循环): 算法 2(嵌套 for 循环): 大 O 表示 时间复杂度分析 关注执行最多的一段代 ...

  5. C++ 指针的引用和指向引用的指针

    指向引用的指针 简单使用指针的一个例子就是: int a = 1; int *p = &a; 预先强调: 没有指向引用的指针 原因: 因为引用 不是对象,没有地址. 但是指向引用的指针是什么形 ...

  6. JMeter学习笔记--JDBC测试计划-连接Mysql

    1.首先要下载jar包,mysql-connector-java-5.1.7-bin.jar 放到Jmeter的lib文件下ext下 2.添加JDBC Connection Configuration ...

  7. 关于React采坑(憨批)系列---类组件(class MyCom extends React.Component--VM47:9 Uncaught TypeError: Super expression must either be null or a function, not undefined)

    今天在学习React中的类组件时,突然给我报错VM47:9 Uncaught TypeError: Super expression must either be null or a function ...

  8. vuex基础(vuex基本结构与调用)

    import Vue from 'vue'; import Vuex from 'vuex'; Vue.use(Vuex); const modulesA = { state:{//状态 count: ...

  9. Django笔记&教程 2-2 URL详细匹配规则

    Django 自学笔记兼学习教程第2章第2节--URL详细匹配规则 点击查看教程总目录 本章第一节中我们简单介绍了URL与View关系 简单概括来说,网页请求的url会通过urls.py里面的urlp ...

  10. [noi110]翘课

    发现加边操作不好处理,因此考虑先加完所有边后删边. 删去一对边x到y,如果两者中有一个不翘课显然没有意义,那么如果都翘课了那么就对他们进行判断,如果无法翘课就继续搜下去. 这样的时间复杂度看上去似乎是 ...