luogu3376 【模板】 网络最大流
题目大意
给出一个网络图,以及其源点和汇点,求出其网络最大流。
概念
可以把网络图看作管道,节点看作管道的交界处。流就像是管道里的流水。管道有个容量(相当于横截面积),还会有个流量(相当于水流占了管道的多少横截面积)。管道的交界处的流量满足流入的流量和等于流出的流量和。网络图有源点和汇点,水流必须从源流到汇。现在要你求整个图中的最大流(也就是从源点流出的流量的最大值)。
增广路径
在当前水流流动的情况下,对于一个从源点到汇点的路径,如果组成该路径的边都没有满流,则该路径便是一个增广路径。
显然,当前图的水流流动的情况下不存在增广路径当且仅当当前图中的总流量为图的最大流。
求最大流:Dinic算法
总体思路
不断地将流量尽量大的水流注入管道,直到网络图中没有增广路径为止。
具体过程
图中每条边存储的是当前边的剩余容量。对以下操作循环:首先从源点Bfs,对当前的可增广路径进行搜索,设置每一个节点的Level。节点的Level表示水流在网络图中流经的节点的顺序。随后Dfs,根据Bfs所得的Level的顺序向当前可增广路径中注入流量尽可能大的水流。
此处有个问题,处理当前可增广路径后的水流状态不一定就是最大流的水流状态呀!所以我们还要对每一条边设置一条反向边,以达到给机会后悔的目的,其初值为0。Dfs求出当前节点的下一个节点用去的水流nextTake后,当前边剩余容量-=nextTake,其反向边的容量+=nextTake。
最后,当Bfs后,如果汇点的Level没有设置上,意味着不存在增广路径了,此时输出结果:每次循环所得的汇点流出的流量之和。
优化
对于每一个节点我们可以设置一个DfsFrom,每次Dfs时,从DfsFrom开始遍历。具体步骤看代码。注意Dfs中for括号里和循环后对DfsFrom的处理。
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <queue>
using namespace std; const int MAX_NODE = 10010, MAX_EDGE = 100010 * 2, INF=0x3f3f3f3f;
#define LOOP(i, n) for(int i=1; i<=n; i++) struct Dinic
{
private:
struct Node;
struct Edge; struct Node
{
int Level, Id;
Edge *Head, *DfsFrom;
}_nodes[MAX_NODE], *Start, *Target;
int _vCount; struct Edge
{
Node *From, *To;
Edge *Next, *Rev;
int Cap;
}*_edges[MAX_EDGE];
int _eCount; Edge *NewEdge()
{
return _edges[++_eCount] = new Edge();
} Edge *AddEdge(Node *from, Node *to, int cap)
{
Edge *e = NewEdge();
e->From = from;
e->To = to;
e->Cap = cap;
e->Next = from->Head;
from->Head = e;
return e;
} bool Bfs()
{
LOOP(i, _vCount)
_nodes[i].Level = 0;
Start->Level = 1;
static queue<Node*> q;
q.push(Start);
while (!q.empty())
{
Node *cur = q.front();
q.pop();
for (Edge *e = cur->Head; e; e = e->Next)
{
if (!e->To->Level && e->Cap)
{
e->To->Level = cur->Level + 1;
q.push(e->To);
}
}
}
return Target->Level;
} int Dfs(Node *cur, int limit)
{
if (cur == Target)
return limit;
if (limit == 0)
return 0;
int curTake = 0;
for (Edge *e = cur->DfsFrom; e; e = cur->DfsFrom = e->Next)
{
if (e->Cap && e->To->Level == cur->Level + 1)
{
int nextTake = Dfs(e->To, min(limit - curTake, e->Cap));//易忘点:-curTake
e->Cap -= nextTake;
e->Rev->Cap += nextTake;
curTake += nextTake;
}
if (limit - curTake == 0)//易忘点
break;
}
return curTake;
} public:
void Build(int uId, int vId, int cap)
{
Node *u = _nodes + uId, *v = _nodes + vId;
u->Id = uId;
v->Id = vId;
Edge *e1 = AddEdge(u, v, cap), *e2 = AddEdge(v, u, 0);
e1->Rev = e2;
e2->Rev = e1;
} int Proceed()
{
int ans = 0;
while (Bfs())
{
LOOP(i, _vCount)
_nodes[i].DfsFrom = _nodes[i].Head;
ans += Dfs(Start, INF);
}
return ans;
} Dinic() {} Dinic(int totNode, int sId, int tId):_vCount(totNode),Start(_nodes+sId),Target(_nodes+tId)
{
memset(_nodes, 0, sizeof(_nodes));
memset(_edges, 0, sizeof(_edges));
} }; int main()
{
//g.Test();
#ifdef _DEBUG
freopen("c:\\noi\\source\\input.txt", "r", stdin);
#endif
int m, vCount, sId, tId;
scanf("%d%d%d%d", &vCount, &m, &sId, &tId);
static Dinic g(vCount, sId, tId);
int uId, vId, cap;
for (int i = 0; i < m; i++)
{
scanf("%d%d%d", &uId, &vId, &cap);
g.Build(uId, vId, cap);
}
int ans;
ans = g.Proceed();
printf("%d\n", ans);
}
luogu3376 【模板】 网络最大流的更多相关文章
- 【洛谷 p3376】模板-网络最大流(图论)
题目:给出一个网络图,以及其源点和汇点,求出其网络最大流. 解法:网络流Dinic算法. 1 #include<cstdio> 2 #include<cstdlib> 3 #i ...
- [模板]网络最大流 & 最小费用最大流
我的作业部落有学习资料 可学的知识点 Dinic 模板 #define rg register #define _ 10001 #define INF 2147483647 #define min(x ...
- P3376 [模板] 网络最大流
https://www.luogu.org/blog/ONE-PIECE/wang-lao-liu-jiang-xie-zhi-dinic EK 292ms #include <bits/std ...
- P3376 【模板】网络最大流
P3376 [模板]网络最大流 题目描述 如题,给出一个网络图,以及其源点和汇点,求出其网络最大流. 输入输出格式 输入格式: 第一行包含四个正整数N.M.S.T,分别表示点的个数.有向边的个数.源点 ...
- 洛谷 P3376 【【模板】网络最大流】
题目描述 如题,给出一个网络图,以及其源点和汇点,求出其网络最大流. 输入 第一行包含四个正整数N.M.S.T,分别表示点的个数.有向边的个数.源点序号.汇点序号. 接下来M行每行包含三个正整数ui. ...
- 洛谷 P3376 【模板】网络最大流
题目描述 如题,给出一个网络图,以及其源点和汇点,求出其网络最大流. 输入输出格式 输入格式: 第一行包含四个正整数N.M.S.T,分别表示点的个数.有向边的个数.源点序号.汇点序号. 接下来M行每行 ...
- 洛谷——P3376 【模板】网络最大流
题目描述 如题,给出一个网络图,以及其源点和汇点,求出其网络最大流. 输入输出格式 输入格式: 第一行包含四个正整数N.M.S.T,分别表示点的个数.有向边的个数.源点序号.汇点序号. 接下来M行每行 ...
- P3376 【模板】网络最大流dinic算法
P3376 [模板]网络最大流 题目描述 如题,给出一个网络图,以及其源点和汇点,求出其网络最大流. 输入输出格式 输入格式: 第一行包含四个正整数N.M.S.T,分别表示点的个数.有向边的个数.源点 ...
- 「LuoguP3376」 【模板】网络最大流
题目描述 如题,给出一个网络图,以及其源点和汇点,求出其网络最大流. 输入输出格式 输入格式: 第一行包含四个正整数N.M.S.T,分别表示点的个数.有向边的个数.源点序号.汇点序号. 接下来M行每行 ...
随机推荐
- C - Arrival of the General
Problem description A Ministry for Defense sent a general to inspect the Super Secret Military Squad ...
- CentOS 安装dotNetCore
如果要在CentOS上运行.net Core程序,必须安装.net Core Sdk 具体安装 方法,可以参考微软官方站点说明,非常详细: 1)百度搜索 .Net Core 2)先择CentOS版本: ...
- SQLite-编译指示
SQLite – 编译指示 SQLite编译指示命令(PRAGMA)是一个特殊的命令是用于控制各种环境变量和状态标志在SQLite的环境.编译指示值可以读取,也可以根据需求设置. 语法: 查询当前的编 ...
- hadoop单机环境配置
1.配置一台linux服务器(当前使用CentOS7)及一些基本设置 1.1在wmware下制作一台centos服务器(桥接模式) 并设置静态ip (编辑/etc/sysmconfig ...
- iOS安全策略之HTTPS
1.HTTPS传输流程 2.常用加密算法 3.AFN证书校验策略及核心方法 4.SSL Pinning 5.CA证书申请流程 HTTPS经由超文本传输协议进行通信,但利用SSL/TLS来对数据包进行加 ...
- Memcached 之内存管理与删除机制
一.内存的碎片化 如果用c语言直接 malloc,free 来向操作系统申请和释放内存时,在不断的申请和释放过程中,形成了一些很小的内存片断,无法再利用,这种空闲,但无法利用内存的现象称为内存的碎片化 ...
- [转载]windows下github 出现Permission denied (publickey).解决方法
今天在学习github的时候遇到了一些问题,然后爬了一会,找到了解决方法记录下来,以防忘记,当然能帮助别人最好啦! github教科书传送门:http://www.liaoxuefeng.com/ ...
- SSL/TLS 加密新纪元 - Let's Encrypt
转自: https://linux.cn/article-6565-1.html SSL/TLS 加密新纪元 - Let's Encrypt 根据 Let's Encrypt 官方博客消息,Let's ...
- spi简介(自我理解)
因为项目中用到spi总线,网上看了下资料,总感觉云山雾罩的,就向身边的同事问了下,他给我解释了下, 我现在把自己理解的写下来 spi一共四条线,一条选择线,一条数据线,2条数据线.spi是一对多设备, ...
- EurekaLog是什么鬼?
D的all工程文件打开后,莫名其妙就处于等待打开状态.因为最后一次调整是安装了RO9.0.所以一直怀疑是RO的原因.再加上win7授权问题,安装RO一直不顺当.所以折腾的时间最多. 其他把RO全部卸载 ...