终于把最小费用最大流学会了啊……

各种奇奇怪怪的解释我已经看多了,但在某些大佬的指点下,我终于会了。

原来是个好水的东西。

最小费用最大流是什么?

不可能不知道网络流吧?如果不知道,自行百度去……

费用流就是在每条边添加个费用,设你这条边的流量是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的更多相关文章

  1. NSOJ A fairy tale of the two(最小费用最大流、SPFA版本、ZKW版本)

    n,m<=20,给两个n×m布尔矩阵,每次操作可将第一个矩阵的2个相邻元素互换.输出最少操作次数使得两个矩阵完全一样. 比赛的时候想过按照二分图完美匹配的类似做法构图,不过想到边太多以及卡各种题 ...

  2. poj 2195 二分图带权匹配+最小费用最大流

    题意:有一个矩阵,某些格有人,某些格有房子,每个人可以上下左右移动,问给每个人进一个房子,所有人需要走的距离之和最小是多少. 貌似以前见过很多这样类似的题,都不会,现在知道是用KM算法做了 KM算法目 ...

  3. Libre 6013 「网络流 24 题」负载平衡 (网络流,最小费用最大流)

    Libre 6013 「网络流 24 题」负载平衡 (网络流,最小费用最大流) Description G 公司有n 个沿铁路运输线环形排列的仓库,每个仓库存储的货物数量不等.如何用最少搬运量可以使n ...

  4. 图论算法-最小费用最大流模板【EK;Dinic】

    图论算法-最小费用最大流模板[EK;Dinic] EK模板 const int inf=1000000000; int n,m,s,t; struct node{int v,w,c;}; vector ...

  5. 【Luogu】P3381最小费用最大流模板(SPFA找增广路)

    题目链接 哈  学会最小费用最大流啦 思路是这样. 首先我们有一个贪心策略.如果我们每次找到单位费用和最短的一条增广路,那么显然我们可以把这条路添加到已有的流量里去——不管这条路的流量是多大,反正它能 ...

  6. 费用流+SPFA ||Luogu P3381【模板】最小费用最大流

    题面:[模板]最小费用最大流 代码: #include<cstdio> #include<cstring> #include<iostream> #include& ...

  7. 费用流+SPFA ||【模板】最小费用最大流

    题面:[模板]最小费用最大流 代码: #include<cstdio> #include<cstring> #include<iostream> #include& ...

  8. P3381 【模板】最小费用最大流

    P3381 [模板]最小费用最大流 题目描述 如题,给出一个网络图,以及其源点和汇点,每条边已知其最大流量和单位流量费用,求出其网络最大流和在最大流情况下的最小费用. 输入输出格式 输入格式: 第一行 ...

  9. nyoj 712 探 寻 宝 藏--最小费用最大流

    问题 D: 探 寻 宝 藏 时间限制: 1 Sec  内存限制: 128 MB 题目描述 传说HMH大沙漠中有一个M*N迷宫,里面藏有许多宝物.某天,Dr.Kong找到了迷宫的地图,他发现迷宫内处处有 ...

随机推荐

  1. leetcode-220-存在重复元素③*

    题目描述: 方法一:二叉搜索树+滑动窗口 方法二:桶排序 O(N) class Solution: def containsNearbyAlmostDuplicate(self, nums: List ...

  2. 校园商铺-4店铺注册功能模块-4Dto之ShopExecution的实现

    1. DTO:添加店铺的返回类型 问题:为什么不直接用实体类Shop呢? 原因:在操作Shop的时候,必然会有一个状态.添加店铺,添加成功,还是添加失败? 如果添加失败,失败是一个什么状态,这些都是要 ...

  3. linux中对EINTR错误的处理

    https://www.cnblogs.com/flyfish10000/articles/2576885.html EINTR错误的产生:当阻塞于某个慢系统调用的一个进程捕获某个信号且相应信号处理函 ...

  4. arcmap分类标注问题

    在给图层标注的时候,经常出现冲突后有些标注出不来,需要将某些个别的点要素进行标注位置调整,如下图: 处理步骤如下, (1)打开Maplex标注引擎.从ToolBars中打开Labeling工具,勾选U ...

  5. 关于Unity中的物理

    碰撞器Colliders Unity有两种类型的碰撞体:网格碰撞体(Mesh Colliders)和原始碰撞体(Primitive Colliders). 网格碰撞体组件使用导入的网格数据,可用于环境 ...

  6. 【bzoj 3489】A simple rmq problem

    题目 \(kdt\)就是数点神器 我们先扫两遍处理出每个数上一次出现的位置\(pre_i,nxt_i\),之后变成\((i,pre_i,nxt_i)\)这样一个三维空间上的点 就变成了求一个立方体的最 ...

  7. spring AOP 编程--AspectJ注解方式 (4)

    1. AOP 简介 AOP(Aspect-Oriented Programming, 面向切面编程): 是一种新的方法论, 是对传统 OOP(Object-Oriented Programming, ...

  8. Go前言

    Go语言为并发而生 硬件制造商正在为处理器添加越来越多的内核以来提高性能.所有数据中心都在这些处理器上运行,今天的应用程序使用多个微服务来维护数据库连接,消息队列和维护缓存.所以,开发的软件和编程语言 ...

  9. Linux 容器 vs 虚拟机——谁更胜一筹

    自从Linux上的容器变得流行以来,了解Linux容器和虚拟机之间的区别变得更加棘手.本文将向您提供详细信息,以了解Linux容器和虚拟机之间的差异. Linux容器vs虚拟机 - 应用程序与操作系统 ...

  10. Eclipse Unable to install breakpoint in XXX 解决办法

    Debug 时偶尔会出现:Eclipse Unable to install breakpoint in XXX 情况一: 清除所有断点就行了,原因是断点打到注释上了. breakpoint 窗口: ...