P3376 【模板】网络最大流(luogu)
P3376 【模板】网络最大流(luogu)
最大流的dinic算法模板(采取了多种优化)
| 优化 | 时间 |
|---|---|
| inline+当前弧+炸点+多路增广 | 174ms |
| no 当前弧 | 175ms |
| no 炸点 | 249 |
自己对最大流算法的理解:
通过bfs对剩余图进行分层,剩余图中有流量的边(cap>0)才能bfs,然后dfs找增广路(满足dis[v]==dis[u]+1&&cap(u,v)>0)
各种优化(基本是在dfs(s,flow)上做的)的实现思路:
- 多路增广
通过记录从s流出的各个边增广的和,而不是找到目标t就返回(可以比较一下两段代码
//当前弧优化效果不明显可不加
struct Edge{
int v,nxt,cap;
}es[maxm];
void add(int u,int v,int cap){
//printf("db add %d->%d %d\n",u,v,cap);
es[cnt].v=v,es[cnt].cap=cap,es[cnt].nxt=head[u];
head[u]=cnt,cnt++;
}
int bfs(int s,int des){
CL(dep,-1);
queue<int> q;q.push(s);
dep[s]=1;
while(!q.empty()){
int tmp=q.front();q.pop();
for(int t=head[tmp];t!=-1;t=es[t].nxt){
int v=es[t].v,cap=es[t].cap;
if(cap>0&&dep[v]==-1){//dep[v]==-1代表未访问过
dep[v]=dep[tmp]+1;q.push(v);
if(v==des) return 1;
}
}
}
return dep[des]>-1?1:0;//是否能到达目的节点
}
int dfs(int s,int fl,int des){//fl流入s的流量,向s的出边增光
if(s==des||fl<=0) return fl;
int res=0; //多路增光
for(int t=head[s];fl&&t!=-1;t=es[t].nxt){
int v=es[t].v,cap=es[t].cap;
if(dep[v]==dep[s]+1){
int f=dfs(v,min(fl,cap),des);
es[t].cap-=f,es[t^1].cap+=f,res+=f;
fl-=f;
}
}
if(res==0) dep[s]=-2;//炸点优化
return res;
}
所以朴素的方式往往需要一次bfs,多次dfs;而经过多路扩展则是一次bfs,一次dfs(因为一次dfs就找全了所有增广路)
- 当前弧优化
思想是记录节点s应该流向的边,假设有这样s->1,s->2,s->3假设s->2时流量flow已经耗尽,那么下一次从3流出的应该从s->2开始(注意流向s可能有多条路径,flow只是流向s的某一条路径的流量)
for(int x=cur[s];x!=-1;x=e[x].next){
cur[s]=x;//当前弧优化
if(d[e[x].v]==d[s]+1&&e[x].cap>0){
//....
if(!flow) break;//当前已经没有流量
}
}
- 炸点优化
一个非目的点,流不出任何流量,那么以后都不用向该点流,所以d[s]=-2(以后dfs都不会流向它)
代码
//dinic 算法处理最大流问题
#include<bits/stdc++.h>
using namespace std;
#define inf 0x3f3f3f3f
const int maxn=1e4+10;
const int maxm=1e5+10;
int s,t,n,m;
struct Edge{
int v,next,cap;
}e[maxm<<1];// begin from 0
int head[maxn];//-1 init
int cnt=-1;
int d[maxn];//init inf
int que[maxn<<2];
int cur[maxn];//for 当前弧优化
inline void add(int u,int v,int cap){//边从0开始
//e[cnt]=(Edge){v,head[u],cap};
cnt++;
e[cnt].v=v,e[cnt].next=head[u],e[cnt].cap=cap;
head[u]=cnt;
}
inline bool bfs(int ss,int tt){
memset(d,inf,sizeof(d));
memcpy(cur,head,sizeof(head));
int h=0,t=1;
que[0]=ss,d[ss]=1;
while(t>h){
int u=que[h];
h++;
for(int x=head[u];x!=-1;x=e[x].next){
int v=e[x].v;
if(d[v]>=inf&&e[x].cap>0){
d[v]=d[u]+1;//d[]用作vis[]
que[t++]=v;
}
}
}
//printf("db %d\n",d[tt]);
return d[tt]<inf;
}
inline int dfs(int s,int flow){
if(s==t||flow<=0) return flow;
int res=0;
for(int x=cur[s];x!=-1;x=e[x].next){
cur[s]=x;//当前弧优化
if(d[e[x].v]==d[s]+1&&e[x].cap>0){
int f=dfs(e[x].v,min(flow,e[x].cap));
e[x].cap-=f;
e[x^1].cap+=f;
res+=f;//通过res记录多路扩展
flow-=f;
if(!flow) break;//当前已经没有流量
}
}
if(!res) d[s]=-2;//炸点优化,s没能流出任何流量,那么本次bfs下的dfs无需再向s流
return res;
}
inline void dbgraph(){
for(int i=1;i<=n;i++){
printf("%d:",i);
for(int x=head[i];x!=-1;x=e[x].next){
if(e[x].cap>0) printf("%d\t",e[x].v);
}
printf("\n");
}
}
int main(){
//freopen("in.txt","r",stdin);
scanf("%d %d %d %d",&n,&m,&s,&t);
memset(head,-1,sizeof(head));
for(int i=1;i<=m;i++){
int u,v,cap;
scanf("%d %d %d",&u,&v,&cap);
add(u,v,cap);
add(v,u,0);
}
//dbgraph();
int ans=0;
while(bfs(s,t)){
//printf("in bfs\n");
ans+=dfs(s,inf);
}
printf("%d\n",ans);
//fclose(stdin);
return 0;
}
补充一点
从EK算法到Dinic算法有很重要的原因在于下面一张图

P3376 【模板】网络最大流(luogu)的更多相关文章
- P3376 [模板] 网络最大流
https://www.luogu.org/blog/ONE-PIECE/wang-lao-liu-jiang-xie-zhi-dinic EK 292ms #include <bits/std ...
- 【洛谷 p3376】模板-网络最大流(图论)
题目:给出一个网络图,以及其源点和汇点,求出其网络最大流. 解法:网络流Dinic算法. 1 #include<cstdio> 2 #include<cstdlib> 3 #i ...
- 【Luogu P3376】网络最大流
Luogu P3376 最大流是网络流模型的一个基础问题. 网络流模型就是一种特殊的有向图. 概念: 源点:提供流的节点(入度为0),类比成为一个无限放水的水厂 汇点:接受流的节点(出度为0),类比成 ...
- [模板]网络最大流 & 最小费用最大流
我的作业部落有学习资料 可学的知识点 Dinic 模板 #define rg register #define _ 10001 #define INF 2147483647 #define min(x ...
- Dinic最大流 || Luogu P3376 【模板】网络最大流
题面:[模板]网络最大流 代码: #include<cstring> #include<cstdio> #include<iostream> #define min ...
- luogu P3376 【模板】网络最大流(no)ek
题目描述 如题,给出一个网络图,以及其源点和汇点,求出其网络最大流. 输入输出格式 输入格式: 第一行包含四个正整数N.M.S.T,分别表示点的个数.有向边的个数.源点序号.汇点序号. 接下来M行每行 ...
- P3376 【模板】网络最大流
P3376 [模板]网络最大流 题目描述 如题,给出一个网络图,以及其源点和汇点,求出其网络最大流. 输入输出格式 输入格式: 第一行包含四个正整数N.M.S.T,分别表示点的个数.有向边的个数.源点 ...
- P3376 【模板】网络最大流dinic算法
P3376 [模板]网络最大流 题目描述 如题,给出一个网络图,以及其源点和汇点,求出其网络最大流. 输入输出格式 输入格式: 第一行包含四个正整数N.M.S.T,分别表示点的个数.有向边的个数.源点 ...
- 『题解』洛谷P3376 【模板】网络最大流
Problem Portal Portal1:Luogu Description 如题,给出一个网络图,以及其源点和汇点,求出其网络最大流. Input 第一行包含四个正整数\(N,M,S,T\),分 ...
随机推荐
- jmeter出现乱码怎么解决
本文是抄袭安大叔的性能 如果想在性能获得更好的发展 请添加公众号:测试那点事 大叔的群号:435092293 大叔曾经担任百度技术总监 很牛逼 相信大叔知道了不会怪我 毕竟我是你的学 ...
- Python数据分析-----数据分类
1.常见的分类算法主要有: (1)KNN算法 (2)贝叶斯方法 (3)决策树 (4)人工神经网络 (5)支持向量机(SVM) 2.KNN算法 (1)KNN应用场景: 比方说样本中有很多零食.很多电器. ...
- kernel相关前沿技术了解
Intel MPX https://en.wikipedia.org/wiki/Intel_MPX SPDK http://aidaiz.com/spdk/ TCMalloc http://goog- ...
- win10卸载瑞星
下载了一个软件,没有注意就不小心装上了瑞星这个流氓软件 百度N种办法并不能解决~ 我试过正常卸载.试过WIN自带卸载.试过重装再卸载 最后采取最傻瓜最暴力的办法 ctrl+alt+delete 打开任 ...
- css 垂直居中方法总结
工作中遇到垂直居中问题,特此总结了一下几种方式与大家分享.本文讨论的垂直居中仅支持IE8+ 1.使用绝对定位垂直居中 HTML <div class="container"& ...
- Gradle学习总结——抓重点学Gradle
前言 网上关于Gradle的教程很多,但很多都是以"面"切入- 通过大量讲解其用法及其API分类来阐述.但Gradle API使用技巧众多,API更是成千上百,臣妾记不住呀.个人深 ...
- 2015 Multi-University Training Contest 10 hdu 5411 CRB and Puzzle
CRB and Puzzle Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others)To ...
- ZOJ 2314 Reactor Cooling
Reactor Cooling Time Limit: 5000ms Memory Limit: 32768KB This problem will be judged on ZJU. Origina ...
- (5)全局异常捕捉【从零开始学Spring Boot】
在一个项目中的异常我们我们都会统一进行处理的,那么如何进行统一进行处理呢? 新建一个类GlobalDefaultExceptionHandler, 在class注解上@ControllerAdvice ...
- 深入MNIST code测试
本系列文章由 @yhl_leo 出品,转载请注明出处. 文章链接: http://blog.csdn.net/yhl_leo/article/details/50624471 依照教程:深入MNIST ...