今天学习了最小费用最大流,是网络流算法之一。可以对于一个每条边有一个容量和一个费用(即每单位流的消耗)的图指定一个源点和汇点,求在从源点到汇点的流量最大的前提下的最小费用。

  这里讲一种最基础也是最好掌握的实现算法,就是spfa求费用流。

  其实也很简单,在最大流的基础上,我们将dfs增广替换成对于费用为权值的边跑spfa得到的最短路,相当于一个贪心的思想。证明有一定难度,稍微口糊一下,就像ford-fulkerson一样,这个算法每次都能找到一条单位流费用和最小的路径,又由于路径中每条边的流量相等,每次增广就能使得单位流量的平均费用更小,而最大流流量是不变的,这样就能使得费用最小。

  网络流算法主要考建图,所以模板只要会默下来就够了,还有就是:

  一定要会写spfa!!!

  对于spfa,这里有两个小优化。

  一个是slf优化,就是对于spfa的进队操作,进之前判一下若小于队头就直接插在队头,这个还是蛮有用的,可以提升点速度;

  另一个是记路径的时候可以不用记前趋点,只要记边,反向边指向的就是前趋,就不用多开一个数组的空间。(这个是zxyer学长当时说的)

  下面贴上代码:

  这是网络流24题中的餐巾计划问题。

  题目大意是:一个饭馆每天需要使用ri条干净的餐巾,用完就脏了,干净餐巾可以由3种方式得到:

    1.直接购买,p元一条;

    2.快洗,需要t1天,花费w1元;

    3.慢洗,需要t2天,花费w2元;

  所以我们建6种边,把每天拆成两个点,分别为干净的(1~n)和脏的(n+1~n*2),这里要注意的是,干净的不直接向脏的连边,而是连向T,相当于必须送满ri条给顾客使用,再从S送到脏的一边。

 #include<cstdio>
using namespace std;
const int inf=;
int n,m,p,t1,w1,t2,w2,tot=,mx,q[],d[],pree[],h[];
struct edge{int to,nxt,cst,cap;}e[];
bool vis[];
int mn(int x,int y){return x>y?y:x;}
void add(int fr,int to,int cst,int cap)
{
e[++tot]={to,h[fr],cst,cap};h[fr]=tot;
e[++tot]={fr,h[to],-cst,};h[to]=tot;//建一条容量为0费用为-cap的反向边且满足正反边异或值为1方便将边取反
}
void init()
{
scanf("%d%d%d%d%d%d",&n,&p,&t1,&w1,&t2,&w2);
for(int i=;i<=n;i++)
{
int r;scanf("%d",&r);mx+=r;
add(,n+i,p,inf);//直接购买花费p元
add(,i,,r);//当天用完的旧餐巾
add(i+n,n*+,,r);//当天需要使用的新餐巾
if(i+t1<=n)add(i,i+n+t1,w1,inf);//快洗花费t1时间,每条w1元
if(i+t2<=n)add(i,i+n+t2,w2,inf);//快洗花费t2时间,每条w2元
if(i<n)add(i,i+,,inf);//不需要做任何事的餐巾积到明天(也可以先洗完然后放在那边等就是n+i连向n+i+1)
}
n=n*+;
}
bool spfa()
{
for(int i=;i<=n;i++)d[i]=inf;
int l=,r=;q[]=;vis[]=;
while(l!=r)
{
int x=q[l=l==n?:l+];
for(int i=h[x];i;i=e[i].nxt)
if(e[i].cap&&e[i].cst+d[x]<d[e[i].to])
{
int v=e[i].to;
pree[v]=i;
d[v]=d[x]+e[i].cst;
if(!vis[v])
{
if(d[v]>d[l+])q[r=r==n?:r+]=v;
else q[l]=v,l=l==?n:l-;//这边加的是slf优化,用了循环队列来写
vis[v]=;
}
}
vis[x]=;
}
return d[n]==inf?:;
}
int costflow()
{
int cost=,mm=;
while(spfa())
{
int mi=inf;
for(int i=n;i;i=e[pree[i]^].to)//记路径时记下前趋路径
mi=mn(mi,e[pree[i]].cap);
for(int i=n;i;i=e[pree[i]^].to)
{
int ee=pree[i];
e[ee].cap-=mi;
e[ee^].cap+=mi;
}
cost+=d[n]*mi;
mm+=mi;
}
return mm==mx?cost:;//是否满流的判断(这道题不会出现不满流的情况,所以不加也可)
}
int main()
{
init();
printf("%d",costflow());
return ;
}

初识费用流 模板(spfa+slf优化) 餐巾计划问题的更多相关文章

  1. 【zkw费用流】[网络流24题]餐巾计划问题

    题目描述 一个餐厅在相继的N天里,第i天需要Ri块餐巾(i=l,2,-,N).餐厅可以从三种途径获得餐巾. (1)购买新的餐巾,每块需p分: (2)把用过的餐巾送到快洗部,洗一块需m天,费用需f分(f ...

  2. 【费用流】【Next Array】费用流模板(spfa版)

    #include<cstdio> #include<algorithm> #include<cstring> #include<queue> using ...

  3. HDU 6611 K Subsequence(Dijkstra优化费用流 模板)题解

    题意: 有\(n\)个数\(a_1\cdots a_n\),现要你给出\(k\)个不相交的非降子序列,使得和最大. 思路: 费用流建图,每个点拆点,费用为\(-a[i]\),然后和源点连边,和后面非降 ...

  4. [BZOJ 2200][Usaco2011 Jan]道路和航线 spfa+SLF优化

    Description Farmer John正在一个新的销售区域对他的牛奶销售方案进行调查.他想把牛奶送到T个城镇 (1 <= T <= 25,000),编号为1T.这些城镇之间通过R条 ...

  5. HDU2686 费用流 模板

    Matrix Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Subm ...

  6. spfa + slf优化

    最近在练习费用流 , 不是要用spfa吗 ,我们教练说:ns学生写朴素的spfa说出去都让人笑 . QwQ,所以就去学了一下优化 . slf优化就是双向队列优化一下,本来想用lll优化,可是优化后我t ...

  7. hdu1533 费用流模板

    Going Home Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total ...

  8. POJ - 3268 Silver Cow Party SPFA+SLF优化 单源起点终点最短路

    Silver Cow Party One cow from each of N farms (1 ≤ N ≤ 1000) conveniently numbered 1..N is going to ...

  9. 费用流模板(带权二分图匹配)——hdu1533

    /* 带权二分图匹配 用费用流求,增加源点s 和 汇点t */ #include<bits/stdc++.h> using namespace std; #define maxn 1000 ...

随机推荐

  1. QObject类 moc处理后代码

    QObject在QT中是所有类的基类,经过MOC处理后代码如下 之所以贴出这段代码,是因为很多流程追踪到最后一些关键性函数都是出自这个类 源码 4.8.6 MOC版本 63 /************ ...

  2. lucene 学习之编码篇

    本文环境:lucene5.2     JDK1.7   IKAnalyzer 引入lucene相关包 <!-- lucene核心包 --> <dependency> <g ...

  3. html5 canvas 图像处理

    1.图像放大缩小 <script> var cvs = document.getElementById("canvas"); cvs.width = cvs.heigh ...

  4. 【codevs1404】字符串匹配 KMP

    题目描述 给你两个串A,B,可以得到从A的任意位开始的子串和B匹配的长度.给定K个询问,对于每个询问给定一个x,求出匹配长度恰为x的位置有多少个.N,M,K<=200000 输入 第一行三个数 ...

  5. 使用for循环遍历数组元素

    循环可以将代码块执行指定的次数.如果您希望一遍又一遍地运行相同的代码,并且每次的值都不同,那么使用循环是很方便的.迭代语句又叫循环语句. JavaScript 支持不同类型的循环: for - 循环代 ...

  6. 分离IE9以下浏览器

    /*判断浏览器版本是否过低*/ function IETester(userAgent) { var UA = userAgent || navigator.userAgent; if (/msie/ ...

  7. python 深浅copy的例子

    1. copy.copy 浅拷贝 只拷贝父对象,不会拷贝对象的内部的子对象.2. copy.deepcopy 深拷贝 拷贝对象及其子对象一个很好的例子:import copya = [1, 2, 3, ...

  8. [洛谷P4781]【模板】拉格朗日插值

    题目大意:给你$n(n\leqslant2000)$个点,要你求$n-1$次经过这$n$个点的多项式在$k$处的值 题解:$Lagrange$插值:$$f_x=\sum\limits_{i=1}^ky ...

  9. POJ3261:Milk Patterns——题解

    http://poj.org/problem?id=3261 给一个序列,求至少出现 k 次的最长重复子串,这 k 个子串可以重叠. 论文题+傻逼题. 上一道题(POJ1743)会做即可. 还是二分长 ...

  10. 51NOD 1594:Gcd and Phi——题解

    http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1594 参考及详细推导:http://www.cnblogs.com/ri ...