序:
在之前的文章中实现了不利用STL实现EK算法,效率也较高。这次我们企图简化代码,减少变量的使用与手写模拟的代码。

注意:vector等STL的container在不开O2优化的时候实现同一个效果普遍比手写要慢。


源代码如下:

/*
About: Max_flow_EK_vector
Auther: kongse_qi
Date:2017/04/22
*/
#include <bits/stdc++.h>
#define INF 0x3f3f3f
#define maxn 20005
#define maxm 200005
#define g_t Get_Int()
#define p_b(x) push_back(x)
#define read(x) g_t, x = in
using namespace std;
/* 部分宏是为了简洁好看 */ struct Edge
{
int from;
int to;
int weight;
Edge(){}
Edge(int f, int t, int w):
from(f), to(t), weight(w){}
};//边表 vector<Edge> edge;//存所有弧的信息
vector<int> arc[maxn];//每个节点相连的弧的编号
typedef vector<int>::iterator iterator_t;//迭代器
int n, m, st, en, max_flow;
iterator_t pre[maxn];//记录上一条弧的编号便于回溯
char *X, *T, *Buffer, c;//读入优化
int in;
bool flag = true; void Get_All()
{
fseek(stdin, 0, SEEK_END);
int file_lenth = ftell(stdin);
Buffer = (char*)malloc(file_lenth);
rewind(stdin);
T = (X = Buffer)+fread(Buffer,1, file_lenth, stdin);
c = *X;
} void Get_Int()
{
in = 0;
while(c < '0' || c > '9') c = *++X;
while(c >= '0' && c <= '9')
{
in = in*10+c-'0';
c = *++X;
}
return ;
} void Init()//当多组数据的时候
{
edge.resize(0);
for(unsigned i = 0; i != n*2; ++i)
{
arc[i].resize(0);
}
max_flow = 0;
return ;
} void Add_Edge(int st, int en, int weight)
{
edge.p_b(Edge(st, en, weight));
edge.p_b(Edge(en, st, 0));//该弧信息与反向弧
arc[st].p_b(edge.size()-2);//编号
arc[en].p_b(edge.size()-1);
return ;
} void Read()
{
int a, b, c;
read(n), read(m);//见宏定义
read(st), read(en);
for(unsigned i = 0; i != m; ++i)
{
read(a), read(b), read(c);
Add_Edge(a, b, c);
}
return ;
} bool Bfs(int st, int en)
{
queue<int> q;
int cur, ne, we;
bool vis[maxn];
memset(vis, 0, sizeof vis);
q.push(st);
vis[st] = true;
while(!q.empty())
{
cur = q.front(), q.pop();
for(iterator_t i = arc[cur].begin(); i != arc[cur].end(); ++i)//迭代器遍历vector
{
ne = edge[*i].to;
we = edge[*i].weight;
if(vis[ne] == false && we > 0)
{
q.push(ne);
pre[ne] = i;//记录当前的弧编号
vis[ne] = true;
pre[ne] = i;
if(ne == en) return true;
}
}
}
return false;
} void EK(int st, int en)
{
int minn;
while(Bfs(st, en))
{
minn = INF;
for(unsigned i = en; i != st; i = edge[*pre[i]].from)
{
minn = min(minn, edge[*pre[i]].weight);
}
for(unsigned i = en; i != st; i = edge[*pre[i]].from)
{
edge[*pre[i]].weight -= minn;
edge[*pre[i]^1].weight += minn;//由于弧和它的反向弧在读入的时候是两两一起的,故可以用异或得到它反向弧的编号
}
max_flow += minn;
}
return ;
} int main()
{
freopen("test.in", "r", stdin); Get_All();
Init();
Read();
EK(st, en);
printf("%d\n", max_flow); return 0;
}

基本意思与非vector实现时是一样的。
效率:非常慢!比非vector实现的要慢1倍,要不是读入优化比较狠肯定要超时。

建议OI比赛的时候不使用STL容器,除非使用例如map等极难手写实现的容器。
不然就等着后面的大数据TLE吧。

箜瑟_qi 2017.04.22 22:14

vector实现最大流EK算法的更多相关文章

  1. 二分图的最大匹配——最大流EK算法

    序: 既然是个图,并且求边数的最大值.那么这就可以转化为网络流的求最大流问题. 只需要将源点与其中一子集的所有节点相连,汇点与另一子集的所有节点相连,将所有弧的流量限制置为1,那么最大流 == 最大匹 ...

  2. 最大流EK算法/DINIC算法学习

    之前一直觉得很难,没学过网络流,毕竟是基础知识现在重新来看. 定义一下网络流问题,就是在一幅有向图中,每条边有两个属性,一个是cap表示容量,一个是flow 表示流过的流量.我们要求解的问题就是从S点 ...

  3. 最大流——EK算法

    一.算法理论 [基本思想] 反复寻找源点s到汇点t之间的增广路径,若有,找出增广路径上每一段[容量-流量]的最小值delta,若无,则结束.在寻找增广路径时,可以用BFS来找,并且更新残留网络的值(涉 ...

  4. (通俗易懂小白入门)网络流最大流——EK算法

    网络流 网络流是模仿水流解决生活中类似问题的一种方法策略,来看这么一个问题,有一个自来水厂S,它要向目标T提供水量,从S出发有不确定数量和方向的水管,它可能直接到达T或者经过更多的节点的中转,目前确定 ...

  5. POJ-1459(最大流+EK算法)

    Power Network POJ-1459 这题值得思索的就是特殊的输入,如何输入一连串字符.这里采用的方法是根据输入已知的输入格式,事先预定好要接受的数据类型. 这里套用的板子也是最大流的模板,但 ...

  6. 最大流EK算法模板

    最近学了下最大流算法,大概思想算是懵懵懂懂了,现在想把模板记录下来,以备后面深刻学习之用. #include<cstdio> #include<cstring> using n ...

  7. POJ1273 最大流 EK算法

    套了个EK的模板 //#pragma comment(linker, "/STACK:16777216") //for c++ Compiler #include <stdi ...

  8. 【转】最大流EK算法

    转自:http://www.cnblogs.com/kuangbin/archive/2011/07/26/2117636.html 图-1 如图-1所示,在这个运输网络中,源点S和汇点T分别是1,7 ...

  9. 最大流EK算法

    给定一个有向图G=(V,E),把图中的边看作 管道,每条边上有一个权值,表示该管道 的流量上限.给定源点s和汇点t,现在假设 在s处有一个水源,t处有一个蓄水池,问从 s到t的最大水流量是多少? 网络 ...

随机推荐

  1. wcf、web api、webservicer 之间的区别

    名次注解 SOAP 简单对象访问协议(SOAP)是一种轻量的.简单的.基于 XML 的协议,是交换数据的一种协议规范,是一种轻量的.简单的.基于XML(标准通用标记语言下的一个子集)的协议,它被设计成 ...

  2. 10大支持移动“触摸操作”的JavaScript框架

    摘要:移动开发行业的发展速度让人目不暇接,也在此大势之下,推出移动网站App成为开发者必经之路,如何让触屏设备 更易使用?如何让网站对触摸手势做出反应并使触摸更友好?所有这一切,皆因JavaScrip ...

  3. Spark2.1.0分布式集群安装

    一.依赖文件安装 1.1 JDK 参见博文:http://www.cnblogs.com/liugh/p/6623530.html 1.2 Hadoop 参见博文:http://www.cnblogs ...

  4. 解决Appium 抓取toast

    首先我们先看看这个gif,图中需要,要抓取的字符串--->请输入转让份数 1.要导入java-client-5.0.0-SNAPSHOT.jar 包的地址:链接:http://pan.baidu ...

  5. 1.跨平台开发之~ VSCode开发第一个C程序

    VSCode的安装就不讲了,可以参考这个(http://www.cnblogs.com/dunitian/p/6661644.html) 写一个简单的C,然后F5运行,根据提示来配置文件 删掉前面的内 ...

  6. MySQL flashback 功能

    1. 简介 mysqlbinlog flashback(闪回)用于快速恢复由于误操作丢失的数据.在DBA误操作时,可以把数据库恢复到以前某个时间点(或者说某个binlog的某个pos).比如忘了带wh ...

  7. TextView的几个属性

    1. android:autoLink 自动识别文本中包含的链接,如网址.邮箱.电话.地图等:属性值有如下几种: web------------ ------只识别网址 email---------- ...

  8. 使用assets目录来实现插件机制

    /** * 管理接口. * @author jevan * @version 1.0 at 2013-12-6 * */ public interface IManage { /** * 注册平台接口 ...

  9. Python入门(一):PTVS写Python程序,调试模式下input()提示文字乱码问题

    前两天写了Python入门(一),里面提到,使用VS2013+PTVS进行Python开发. 就在准备为第二篇写个demo的时候,发现了一个问题,各种解决无果,有些纠结 Python中输入函数是inp ...

  10. 免费视频播放器videojs中文教程

    Video.js是一款web视频播放器,支持html5和flash两种播放方式.更多关于video.js的介绍,可以访问官方网站介绍,我之前也写过一篇关于video.js的使用心得,有兴趣的可以点这里 ...