最小费用最大流——EK+SPFA
终于把最小费用最大流学会了啊……
各种奇奇怪怪的解释我已经看多了,但在某些大佬的指点下,我终于会了。
原来是个好水的东西。
最小费用最大流是什么?
不可能不知道网络流吧?如果不知道,自行百度去……
费用流就是在每条边添加个费用,设你这条边的流量是f" role="presentation">ff,费用为w" role="presentation">ww,则总费用为fw" role="presentation">fwfw。
举个例子,就像是有许多点的一张图,有很多个管子相连,每个管子都有个容量,并且每流一流量就要花一些费用,问总费用最少是多少。
最小费用最大流怎么做?
首先要知道EK算法……
开玩笑的,其实根本不用,我还没打过普通的EK呢,就只是打过dinic和sap。
这个做法其实很简单:
1、用spfa从原点跑最短路,边权为费用(满流的边不用跑)。
2、将最短路径抽出,在上面找一个残余容量最小的。
3、这一路上的残余容量减少,反向弧的容量增加(反向弧的费用为负的正向的费用)。
4、回到第一步,继续做下去,直到从原点跑不到汇点。
这样就可以算出来了,这就是最普通的EK+spfa做法。
当然还有更好的,但我还不会……
时间复杂度我不知道,毕竟,网络流的时间总是很玄学啊……
代码
using namespace std;
#include <cstdio>
#include <cstring>
#include <algorithm>
int n,m,S,T;
struct EDGE
{
int to,c,w;
EDGE *las;
} e[100001];
int ne;
EDGE *last[5001];
#define rev(ei) (e+(int((ei)-e)^1))
int q[1000001];
bool inq[5001];
bool SPFA();
int dis[5001];
EDGE *pre[5001];
void flow(int&,int&);
int main()
{
scanf("%d%d%d%d",&n,&m,&S,&T);
for (int i=1;i<=m;++i)
{
int u,v,c,w;
scanf("%d%d%d%d",&u,&v,&c,&w);
e[ne]={v,c,w,last[u]};
last[u]=e+ne;
++ne;
e[ne]={u,0,-w,last[v]};
last[v]=e+ne;
++ne;
}
int maxflow,mincost;
flow(maxflow,mincost);
printf("%d %d\n",maxflow,mincost);
return 0;
}
bool SPFA()
{
int h=-1,t=0;
memset(dis,127,sizeof dis);
dis[S]=0;
pre[S]=NULL;
q[0]=S;
inq[S]=1;
do
{
++h;
for (EDGE *ei=last[q[h]];ei;ei=ei->las)
if (ei->c && dis[q[h]]+ei->w<dis[ei->to])
{
dis[ei->to]=dis[q[h]]+ei->w;
pre[ei->to]=ei;
if (!inq[ei->to])
{
inq[ei->to]=1;
q[++t]=ei->to;
}
}
inq[q[h]]=0;
}
while (h!=t);
return dis[T]!=0x7f7f7f7f;
}
void flow(int &maxflow,int &mincost)
{
maxflow=0;
mincost=0;
while (SPFA())
{
int minc=0x7f7f7f7f;
for (EDGE *ei=pre[T];ei;ei=pre[rev(ei)->to])
minc=min(minc,ei->c);
maxflow+=minc;
mincost+=dis[T]*minc;
for (EDGE *ei=pre[T];ei;ei=pre[rev(ei)->to])
{
ei->c-=minc;
rev(ei)->c+=minc;
}
}
}
这个代码我没有对过任何的标程,相信各位可以凭借自己的理解打出来,解析就不打了。
最小费用最大流——EK+SPFA的更多相关文章
- NSOJ A fairy tale of the two(最小费用最大流、SPFA版本、ZKW版本)
n,m<=20,给两个n×m布尔矩阵,每次操作可将第一个矩阵的2个相邻元素互换.输出最少操作次数使得两个矩阵完全一样. 比赛的时候想过按照二分图完美匹配的类似做法构图,不过想到边太多以及卡各种题 ...
- poj 2195 二分图带权匹配+最小费用最大流
题意:有一个矩阵,某些格有人,某些格有房子,每个人可以上下左右移动,问给每个人进一个房子,所有人需要走的距离之和最小是多少. 貌似以前见过很多这样类似的题,都不会,现在知道是用KM算法做了 KM算法目 ...
- Libre 6013 「网络流 24 题」负载平衡 (网络流,最小费用最大流)
Libre 6013 「网络流 24 题」负载平衡 (网络流,最小费用最大流) Description G 公司有n 个沿铁路运输线环形排列的仓库,每个仓库存储的货物数量不等.如何用最少搬运量可以使n ...
- 图论算法-最小费用最大流模板【EK;Dinic】
图论算法-最小费用最大流模板[EK;Dinic] EK模板 const int inf=1000000000; int n,m,s,t; struct node{int v,w,c;}; vector ...
- 【Luogu】P3381最小费用最大流模板(SPFA找增广路)
题目链接 哈 学会最小费用最大流啦 思路是这样. 首先我们有一个贪心策略.如果我们每次找到单位费用和最短的一条增广路,那么显然我们可以把这条路添加到已有的流量里去——不管这条路的流量是多大,反正它能 ...
- 费用流+SPFA ||Luogu P3381【模板】最小费用最大流
题面:[模板]最小费用最大流 代码: #include<cstdio> #include<cstring> #include<iostream> #include& ...
- 费用流+SPFA ||【模板】最小费用最大流
题面:[模板]最小费用最大流 代码: #include<cstdio> #include<cstring> #include<iostream> #include& ...
- P3381 【模板】最小费用最大流
P3381 [模板]最小费用最大流 题目描述 如题,给出一个网络图,以及其源点和汇点,每条边已知其最大流量和单位流量费用,求出其网络最大流和在最大流情况下的最小费用. 输入输出格式 输入格式: 第一行 ...
- nyoj 712 探 寻 宝 藏--最小费用最大流
问题 D: 探 寻 宝 藏 时间限制: 1 Sec 内存限制: 128 MB 题目描述 传说HMH大沙漠中有一个M*N迷宫,里面藏有许多宝物.某天,Dr.Kong找到了迷宫的地图,他发现迷宫内处处有 ...
随机推荐
- day33 序列类型,绑定方法,类方法,静态方法,封装继承和多态
Python之路,Day20 = 序列类型,绑定方法,类方法,静态方法,封装继承和多态 序列是指有序的队列,重点在"有序". 一.Python中序列的分类 Python中的序列主要 ...
- csps模拟8990部分题解
题面: 666: 重点在题意转化:每个数可以乘k,代价为k,可以减一,代价为1, 所以跑最短路即可 #include<iostream> #include<cstdio> #i ...
- 边双联通分量缩点+树的直径——cf1000E
题意理解了就很好做 题意:给一张无向图,任意取两个点s,t,s->t的路径上必经边数量为k 求这样的s,t,使得k最大 #include<bits/stdc++.h> #define ...
- 大数据之hadoop集群安全模式
集群安全模式1.概述(1)NameNode启动 NameNode启动时,首先将镜像文件(Fsimage)载入内存,并执行编辑日志(Edits)中的各项操作.-旦在内存中成功建立文件系统元数据的影像,则 ...
- duilib教程之duilib入门简明教程16.结合win32和MFC
虽然duilib自带在MFC中使用duilib的Demo,但只是MFC窗口和duilib窗口不重叠的情况.如果要在MFC窗口中嵌入duilib控件,或者在duilib控件中嵌入MFC的控件的话,就没有 ...
- POJ 3134 - Power Calculus
迭代加深 //Twenty #include<cstdio> #include<cstdlib> #include<iostream> #include<al ...
- RPC远程过程调用实例详解
1.创建IDL文件,定义接口. IDL文件可以由uuidgen.exe创建. 首先找到系统中uuidgen.exe的位置,如:C:\Program Files\Microsoft Visual Stu ...
- 1.RCC of STM32F103
开发板用的是8Mhz晶振,即HSE=8,000,000 f103最大支持72Mhz,即SYSCLK=PLLCLK=72Mhz,HSE=8M,所以PLLM=9即可. /* HCLK = SYSCLK * ...
- char类型在传参时接收不到数据的原因
mybatis的原因!!!!! 数据库这个样子 在postman中调用接口:SQL select * from T_TRAIN_MARSHALLING where TRAIN_NUM is null ...
- iOS开发系列-线程状态
概述 线程从创建到销毁中间存在很多种状态. 线程的状态 通过NSThread创建一条线程,开发者需要负责线程的创建和执行,线程的销毁由系统决定.创建一个继承NSThread的FMThread类,重写d ...