题目链接

https://www.luogu.com.cn/problem/P3376

题目大意

输入格式

第一行包含四个正整数 \(n,m,s,t\),分别表示点的个数、有向边的个数、源点序号、汇点序号。

接下来\(m\)行,每行包含三个正整数 \(u_i,v_i,w_i\),表示第 \(i\) 条有向边从 \(u_i\) 出发,到达 \(v_i\),边权为 \(w_i\)(即该边最大流量为 \(w_i\) )。

输出格式

一行,包含一个正整数,即为该网络的最大流。

题目解析

(待补充,咕咕咕。。。)

参考代码

\(EK\)

#include <bits/stdc++.h>
using namespace std;
const int N = 205;
const long long INF = (1LL << 32);
struct Edge{
int from, to;
long long cap, flow;
};
int n, m, s, t;
long long a[N];
int p[N];
vector <Edge> e;
vector <int> G[N]; void addEdge(int from, int to, int cap, int i)
{
e.push_back((Edge){from, to, cap, 0});
e.push_back((Edge){to, from, 0, 0});
G[from].push_back(i << 1);
G[to].push_back((i << 1)+1);
}
long long maxflow()
{
long long flow = 0;
while (1)
{
memset(a, 0, sizeof(a));
a[s] = INF;
queue <int> q;
q.push(s);
while (!q.empty())
{
int x = q.front();
q.pop();
for (int i=0; i<G[x].size(); ++i)
{
Edge &b = e[G[x][i]];
if (!a[b.to] && b.cap > b.flow)
{
p[b.to] = G[x][i];
a[b.to] = min(a[x], b.cap-b.flow);
q.push(b.to);
}
}
if (a[t]) break;
}
if (!a[t]) break;
for (int u=t; u!=s; u=e[p[u]].from)
{
e[p[u]].flow += a[t];
e[p[u]^1].flow -= a[t];
}
flow += a[t];
}
return flow;
}
int main()
{
int u, v, w;
scanf("%d%d%d%d", &n, &m, &s, &t);
for (int i=0; i<m; ++i)
{
scanf("%d%d%d", &u, &v, &w);
addEdge(u, v, w, i);
}
printf("%lld\n", maxflow());
return 0;
}

\(Dinic\)

#include <bits/stdc++.h>
#define ll long long
using namespace std;
const int INF = 2147483647;
const int N = 205;
struct Edge{
int from, to, cap, flow;
};
int cur[N], depth[N];
int n, m, s, t;
vector <Edge> e;
vector <int> G[N]; void addEdge(int u, int v, int w, int i)
{
e.push_back((Edge){u, v, w, 0});
e.push_back((Edge){v, u, 0, 0});
G[u].push_back(i);
G[v].push_back(i^1);
}
int BFS()
{
queue <int> Q;
memset(depth, 0, sizeof depth);
depth[s] = 1;
Q.push(s);
while (!Q.empty())
{
int u = Q.front(); Q.pop();
for (int i = 0; i < G[u].size(); ++i) {
Edge& b = e[G[u][i]];
if (!depth[b.to] && b.cap > b.flow)
{
depth[b.to] = depth[u] + 1;
Q.push(b.to);
}
}
}
return depth[t];
}
int DFS(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 (depth[b.to] == depth[x]+1 && b.cap > b.flow)
{
if (int c = DFS(b.to, min(a, b.cap - b.flow)))
{
b.flow += c;
e[G[x][i]^1].flow -= c;
flow += c;
a -= c;
if (!a) break;
}
}
}
return flow;
}
ll maxFlow_Dinic()
{
ll ans = 0;
while (BFS()) {
memset(cur, 0, sizeof cur);
ans += DFS(s, INF);
}
return ans;
}
int main()
{
scanf("%d%d%d%d", &n, &m, &s, &t);
for (int i = 0; i < m; ++i) {
int u, v, w;
scanf("%d%d%d", &u, &v, &w);
addEdge(u, v, w, i << 1);
}
printf("%lld\n", maxFlow_Dinic());
return 0;
}

\(ISAP\)

#include <bits/stdc++.h>
#define ll long long
using namespace std;
const int N = 205;
const int INF = 2147483647;
struct Edge {
int from, to, cap, flow;
};
int n, m, s, t, cnt, gap[N], cur[N], dep[N];
vector <int> G[N];
vector <Edge> e; void addEdge(int u, int v, int w, int i)
{
e.push_back((Edge){u, v, w, 0});
e.push_back((Edge){v, u, 0, 0});
G[u].push_back(i);
G[v].push_back(i^1);
}
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] = cnt+1;
++gap[++dep[x]], cur[x] = 0;
return flow;
}
ll maxFlow_ISAP()
{
cnt = n; //Num_of_nodes -> cnt
init();
ll ans = 0;
while (dep[s] <= cnt) ans += augment(s, INF);
return ans;
}
int main()
{
scanf("%d%d%d%d", &n, &m, &s, &t);
for (int i = 0; i < m; i++) {
int u, v, w;
scanf("%d%d%d", &u, &v, &w);
addEdge(u, v, w, i << 1);
}
printf("%lld\n", maxFlow_ISAP());
return 0;
}

感谢支持!

【模板】网络最大流(EK、Dinic、ISAP)(网络流)/洛谷P3376的更多相关文章

  1. 【最大流ISAP】洛谷P3376模板题

    题目描述 如题,给出一个网络图,以及其源点和汇点,求出其网络最大流. 输入输出格式 输入格式: 第一行包含四个正整数N.M.S.T,分别表示点的个数.有向边的个数.源点序号.汇点序号. 接下来M行每行 ...

  2. 网络最大流(EK)

    以前在oi中见到网络流的题都是直接跳过,由于本蒟蒻的理解能力太弱,导致网络流的学习不断推迟甚至被安排在了tarjan之后,原本计划于学习完最短路后就来学网络流的想法也随之破灭,在看完众多大佬 的博客后 ...

  3. 【洛谷 p3376】模板-网络最大流(图论)

    题目:给出一个网络图,以及其源点和汇点,求出其网络最大流. 解法:网络流Dinic算法. 1 #include<cstdio> 2 #include<cstdlib> 3 #i ...

  4. [洛谷P3376题解]网络流(最大流)的实现算法讲解与代码

    [洛谷P3376题解]网络流(最大流)的实现算法讲解与代码 更坏的阅读体验 定义 对于给定的一个网络,有向图中每个的边权表示可以通过的最大流量.假设出发点S水流无限大,求水流到终点T后的最大流量. 起 ...

  5. 洛谷P3376【模板】网络最大流 ISAP

    这篇博客写得非常好呀. 传送门 于是我是DCOI这一届第一个网络流写ISAP的人了,之后不用再被YKK她们嘲笑我用Dinic了!就是这样! 感觉ISAP是会比Dinic快,只分一次层,然后不能增广了再 ...

  6. 洛谷 P3376 【【模板】网络最大流】

    题目描述 如题,给出一个网络图,以及其源点和汇点,求出其网络最大流. 输入 第一行包含四个正整数N.M.S.T,分别表示点的个数.有向边的个数.源点序号.汇点序号. 接下来M行每行包含三个正整数ui. ...

  7. 『题解』洛谷P3376 【模板】网络最大流

    Problem Portal Portal1:Luogu Description 如题,给出一个网络图,以及其源点和汇点,求出其网络最大流. Input 第一行包含四个正整数\(N,M,S,T\),分 ...

  8. 洛谷——P3376 【模板】网络最大流

    题目描述 如题,给出一个网络图,以及其源点和汇点,求出其网络最大流. 输入输出格式 输入格式: 第一行包含四个正整数N.M.S.T,分别表示点的个数.有向边的个数.源点序号.汇点序号. 接下来M行每行 ...

  9. 洛谷 P3376 【模板】网络最大流

    题目描述 如题,给出一个网络图,以及其源点和汇点,求出其网络最大流. 输入输出格式 输入格式: 第一行包含四个正整数N.M.S.T,分别表示点的个数.有向边的个数.源点序号.汇点序号. 接下来M行每行 ...

  10. 洛谷 P3376【模板】网络最大流

    题目描述 如题,给出一个网络图,以及其源点和汇点,求出其网络最大流. 输入输出格式 输入格式: 第一行包含四个正整数N.M.S.T,分别表示点的个数.有向边的个数.源点序号.汇点序号. 接下来M行每行 ...

随机推荐

  1. 洛谷 P2252 [SHOI2002]取石子游戏|【模板】威佐夫博弈

    链接: P2252 [SHOI2002]取石子游戏|[模板]威佐夫博弈 前言: 第一眼大水题,第二眼努力思考,第 N 眼我是大水逼. 题意: 不看题目标题都应该能看出来是取石子类的博弈论. 有两堆石子 ...

  2. 关于linux的fork的一点学习总结

    最近操作系统的实验要用到fork,于是去搜索了一下资料,很幸运地在博客中找到一篇深度好文: http://blog.csdn.net/jason314/article/details/5640969 ...

  3. 字典树(Trie)

    终于学会字典树了,真开心(然后就滚过来写总结了). 首先,字典树到底是个什么东西呢?请看下面这段话: 字典树,常被用来保存与查找大量的字符串,它利用了字符串之间的公共前缀来节约时间,但它的空间花费较大 ...

  4. oeasy教您玩转vim - 56 - # 字符可视化模式

    ​ 可视化编辑 回忆上节课内容 我们学习了关于模式匹配中使用参数 单个参数 :%s/<h2>\(.*\)</h2>/ - \1/g 多个参数 :%s/<img src=\ ...

  5. 从0到1搭建自己的组件(vue-code-view)库(下)

    0x00 前言 书接上文,本文将从源码功能方面讲解下 vue-code-view 组件核心逻辑,您可以了解以下内容: 动态组件的使用. codeMirror插件的使用. 单文件组件(SFC,singl ...

  6. Java 在PPT中创建散点图

    本文将以Java代码示例展示如何在PPT幻灯片中创建散点图表. 创建图表前 需要在Java程序中导入用于操作PPT的jar包 Free Spire.Presentation for Java.可参考如 ...

  7. CentOS 7 tmpwatch 2.11 版本变更,移除 cronjob 任务

    老版本(RHEL6) tmpwatch 原理 在 RHEL6 上,/tmp 目录的清理工作通常是交给 tmpwatch 程序来完成的,tmpwatch 的工作机制是通过 /etc/cron.daily ...

  8. Mybatis3源码加注释后后编译问题

    参考:https://mp.weixin.qq.com/s/v0ihaPsuyGufdc_ImEqX8A给mybatis3源码加注释并编译源代码 编译命令: mvn clean mvn install ...

  9. SpringCloud升级之路2020.0.x版-33. 实现重试、断路器以及线程隔离源码

    本系列代码地址:https://github.com/JoJoTec/spring-cloud-parent 在前面两节,我们梳理了实现 Feign 断路器以及线程隔离的思路,并说明了如何优化目前的负 ...

  10. Python进阶(多线程)

    多线程结构 import threading def worker():#子线程要执行的具体逻辑代码函数 print('threading') t1 = threading.current_threa ...