POJ - 2516 Minimum Cost(最小费用最大流)
1、K种物品,M个供应商,N个收购商。每种物品从一个供应商运送到一个收购商有一个单位运费。每个收购商都需要K种物品中的若干。求满足所有收购商需求的前提下的最小运费。
2、K种物品拆开来,分别对每种物品进行最小费用最大流计算。
建立超级源点和超级汇点:超级源点流向M个供应商,容量为供应商的存储量,费用为0;N个收购商流向超级源点,容量为收购商的需求量,费用为0。
另外,供应商流向收购商,容量为无穷大,费用为对应的单位运费。
3、
1、Bellman-Ford:
#include<iostream>
#include<stdio.h>
#include<vector>
#include<string.h>
#include<queue>
using namespace std; const int maxn=;
const int INF=0x3f3f3f3f; struct Edge{
int from,to,cap,flow,cost;
Edge(int u,int v,int c,int f,int w):from(u),to(v),cap(c),flow(f),cost(w){}
}; struct MCMF{
int n,m;
vector<Edge>edges;
vector<int>G[maxn];
int inq[maxn];//是否在队列中
int d[maxn];//Bellman-Ford
int p[maxn];//上一条弧
int a[maxn];//可改进量 void init(int n){
this->n=n;
for(int i=;i<n;i++)G[i].clear();
edges.clear();
} void AddEdge(int from,int to,int cap,int cost){
edges.push_back(Edge(from,to,cap,,cost));
edges.push_back(Edge(to,from,,,-cost));
m=edges.size();
G[from].push_back(m-);
G[to].push_back(m-);
} bool BellmanFord(int s,int t,int &flow,long long &cost){
for(int i=;i<n;i++)d[i]=INF;
memset(inq,,sizeof(inq));
d[s]=;inq[s]=;p[s]=;a[s]=INF; queue<int>Q;
Q.push(s);
while(!Q.empty()){
int u=Q.front();Q.pop();
inq[u]=;
for(int i=;i<(int)G[u].size();i++){
Edge &e=edges[G[u][i]];
if(e.cap>e.flow&&d[e.to]>d[u]+e.cost){
d[e.to]=d[u]+e.cost;
p[e.to]=G[u][i];
a[e.to]=min(a[u],e.cap-e.flow);
if(!inq[e.to]){Q.push(e.to);inq[e.to]=;}
}
}
}
if(d[t]==INF)return false;
flow+=a[t];
cost+=(long long)d[t]*(long long)a[t];
for(int u=t;u!=s;u=edges[p[u]].from){
edges[p[u]].flow+=a[t];
edges[p[u]^].flow-=a[t];
}
return true;
} //需要保证初始网络中没有负权圈
int MincostMaxflow(int s,int t,long long &cost){
int flow=;cost=;
while(BellmanFord(s,t,flow,cost));
return flow;
}
}MM; int main(){
int N,M,K;//汇点个数,源点个数,物品种类
int need[][];//need[i][j]表示第i个店主需要第j种商品的数量
int totalneed[];//totalneed[i]表示第i种商品的总需求量
int storage[][];//storage[i][j]表示第i个仓库里存储的第j种商品的数量
int totalstorage[];//totalstorage[i]表示第i种商品的总存储量
int cost[][];//cost[i][j]表示当前这个商品从第j个仓库运送到第i个店主的单位价格
bool flag;//false表示存储量不足
long long mincost;//最小花费
int maxflow;//最大流
long long totalcost;
int totalflow;
int S,T;//超级源点,超级汇点 while(~scanf("%d%d%d",&N,&M,&K)){
if(N==&&M==&&K==)break;
memset(totalneed,,sizeof(totalneed));
memset(totalstorage,,sizeof(totalstorage));
flag=true;
totalcost=;
totalflow=; for(int i=;i<N;++i){
for(int j=;j<K;++j){
scanf("%d",&need[i][j]);
totalneed[j]+=need[i][j];
}
}
for(int i=;i<M;++i){
for(int j=;j<K;++j){
scanf("%d",&storage[i][j]);
totalstorage[j]+=storage[i][j];
}
}
for(int i=;i<K;++i){
if(totalneed[i]>totalstorage[i]){
flag=false;
break;
}
} for(int k=;k<K;++k){
MM.init(N+M+);//初始化要放这里
for(int i=;i<N;++i){
for(int j=;j<M;++j){
scanf("%d",&cost[i][j]);
MM.AddEdge(j,M+i,INF,cost[i][j]);//注意加边:j->M+i
}
}
if(flag==false)continue; //超级源点
S=N+M;
for(int v=;v<M;++v){
MM.AddEdge(S,v,storage[v][k],);
}
//超级汇点
T=N+M+;
for(int v=;v<N;++v){
MM.AddEdge(M+v,T,need[v][k],);
} maxflow=MM.MincostMaxflow(S,T,mincost);
if(maxflow<totalneed[k])
flag=false; totalcost+=mincost;
totalflow+=maxflow;
} if(flag==false)printf("-1\n");
else printf("%lld\n",totalcost);
}
return ;
}
2、SPFA版费用流:
/*
SPFA版费用流
最小费用最大流,求最大费用最大流只需要取相反数,结果取相反数即可。
点的总数为N,点的编号0~N-1
*/
#include<iostream>
#include<stdio.h>
#include<string.h>
#include<queue>
using namespace std; const int MAXN=;
const int MAXM=;
const int INF=0x3f3f3f3f;
struct Edge{
int to,next,cap,flow,cost;
}edge[MAXM];
int head[MAXN],tol;
int pre[MAXN],dis[MAXN];
bool vis[MAXN];
int N;//节点总个数,节点编号从0~N-1
void init(int n){
N=n;
tol=;
memset(head,-,sizeof(head));
}
void addedge(int u,int v,int cap,int cost){
edge[tol].to=v;
edge[tol].cap=cap;
edge[tol].cost=cost;
edge[tol].flow=;
edge[tol].next=head[u];
head[u]=tol++;
edge[tol].to=u;
edge[tol].cap=;
edge[tol].cost=-cost;
edge[tol].flow=;
edge[tol].next=head[v];
head[v]=tol++;
}
bool spfa(int s,int t){
queue<int>q;
for(int i=;i<N;i++){
dis[i]=INF;
vis[i]=false;
pre[i]=-;
}
dis[s]=;
vis[s]=true;
q.push(s);
while(!q.empty()){
int u=q.front();
q.pop();
vis[u]=false;
for(int i=head[u];i!=-;i=edge[i].next){
int v=edge[i].to;
if(edge[i].cap>edge[i].flow&&dis[v]>dis[u]+edge[i].cost){
dis[v]=dis[u]+edge[i].cost;
pre[v]=i;
if(!vis[v]){
vis[v]=true;
q.push(v);
}
}
}
}
if(pre[t]==-)return false;
else return true;
}
//返回的是最大流,cost存的是最小费用
int minCostMaxflow(int s,int t,int &cost){
int flow=;
cost=;
while(spfa(s,t)){
int Min=INF;
for(int i=pre[t];i!=-;i=pre[edge[i^].to]){
if(Min>edge[i].cap-edge[i].flow)
Min=edge[i].cap-edge[i].flow;
}
for(int i=pre[t];i!=-;i=pre[edge[i^].to]){
edge[i].flow+=Min;
edge[i^].flow-=Min;
cost+=edge[i].cost*Min;
}
flow+=Min;
}
return flow;
} int main(){
int N,M,K;//汇点个数,源点个数,物品种类
int need[][];//need[i][j]表示第i个店主需要第j种商品的数量
int totalneed[];//totalneed[i]表示第i种商品的总需求量
int storage[][];//storage[i][j]表示第i个仓库里存储的第j种商品的数量
int totalstorage[];//totalstorage[i]表示第i种商品的总存储量
int cost[][];//cost[i][j]表示当前这个商品从第j个仓库运送到第i个店主的单位价格
bool flag;//false表示存储量不足
int mincost;//最小花费
int maxflow;//最大流
int totalcost;
int totalflow;
int S,T;//超级源点,超级汇点 while(~scanf("%d%d%d",&N,&M,&K)){
if(N==&&M==&&K==)break;
memset(totalneed,,sizeof(totalneed));
memset(totalstorage,,sizeof(totalstorage));
flag=true;
totalcost=;
totalflow=; for(int i=;i<N;++i){
for(int j=;j<K;++j){
scanf("%d",&need[i][j]);
totalneed[j]+=need[i][j];
}
}
for(int i=;i<M;++i){
for(int j=;j<K;++j){
scanf("%d",&storage[i][j]);
totalstorage[j]+=storage[i][j];
}
}
for(int i=;i<K;++i){
if(totalneed[i]>totalstorage[i]){
flag=false;
break;
}
} for(int k=;k<K;++k){
init(N+M+);//初始化要放这里
for(int i=;i<N;++i){
for(int j=;j<M;++j){
scanf("%d",&cost[i][j]);
addedge(j,M+i,INF,cost[i][j]);//注意加边:j->M+i
}
}
if(flag==false)continue; //超级源点
S=N+M;
for(int v=;v<M;++v){
addedge(S,v,storage[v][k],);
}
//超级汇点
T=N+M+;
for(int v=;v<N;++v){
addedge(M+v,T,need[v][k],);
} maxflow=minCostMaxflow(S,T,mincost);
if(maxflow<totalneed[k])
flag=false; totalcost+=mincost;
totalflow+=maxflow;
} if(flag==false)printf("-1\n");
else printf("%d\n",totalcost);
}
return ;
}
3、zkw费用流:
/*
zkw费用流
对于二分图类型的比较高效
*/
#include<iostream>
#include<stdio.h>
#include<string.h>
using namespace std; const int MAXN=;
const int MAXM=;
const int INF=0x3f3f3f3f;
struct Edge{
int to,next,cap,flow,cost;
Edge(int _to=,int _next=,int _cap=,int _flow=,int _cost=):
to(_to),next(_next),cap(_cap),flow(_flow),cost(_cost){}
}edge[MAXM];
struct ZKW_MinCostMaxFlow{
int head[MAXN],tot;
int cur[MAXN];
int dis[MAXN];
bool vis[MAXN];
int ss,tt,N;//源点、汇点和点的总个数(编号是0~N-1),不需要额外赋值,调用会直接赋值
int min_cost,max_flow;
void init(){
tot=;
memset(head,-,sizeof(head));
}
void addedge(int u,int v,int cap,int cost){
edge[tot]=Edge(v,head[u],cap,,cost);
head[u]=tot++;
edge[tot]=Edge(u,head[v],,,-cost);
head[v]=tot++;
}
int aug(int u,int flow){
if(u==tt)return flow;
vis[u]=true;
for(int i=cur[u];i!=-;i=edge[i].next){
int v=edge[i].to;
if(edge[i].cap>edge[i].flow&&!vis[v]&&dis[u]==dis[v]+edge[i].cost){
int tmp=aug(v,min(flow,edge[i].cap-edge[i].flow));
edge[i].flow+=tmp;
edge[i^].flow-=tmp;
cur[u]=i;
if(tmp)return tmp;
}
}
return ;
}
bool modify_label(){
int d=INF;
for(int u=;u<N;u++)
if(vis[u])
for(int i=head[u];i!=-;i=edge[i].next){
int v=edge[i].to;
if(edge[i].cap>edge[i].flow&&!vis[v])
d=min(d,dis[v]+edge[i].cost-dis[u]);
}
if(d==INF)return false;
for(int i=;i<N;i++)
if(vis[i]){
vis[i]=false;
dis[i]+=d;
}
return true;
}
/*
直接调用获取最小费用和最大流
输入:start-源点,end-汇点,n-点的总个数(编号从0开始)
返回值:pair<int,int>第一个是最小费用,第二个是最大流
*/
pair<int,int> mincostmaxflow(int start,int end,int n){
ss=start,tt=end,N=n;
min_cost=max_flow=;
for(int i=;i<n;i++)dis[i]=;
while(){
for(int i=;i<n;i++)cur[i]=head[i];
while(){
for(int i=;i<n;i++)vis[i]=false;
int tmp=aug(ss,INF);
if(tmp==)break;
max_flow+=tmp;
min_cost+=tmp*dis[ss];
}
if(!modify_label())break;
}
return make_pair(min_cost,max_flow);
}
}solve; int main(){
int N,M,K;//汇点个数,源点个数,物品种类
int need[][];//need[i][j]表示第i个店主需要第j种商品的数量
int totalneed[];//totalneed[i]表示第i种商品的总需求量
int storage[][];//storage[i][j]表示第i个仓库里存储的第j种商品的数量
int totalstorage[];//totalstorage[i]表示第i种商品的总存储量
int cost[][];//cost[i][j]表示当前这个商品从第j个仓库运送到第i个店主的单位价格
bool flag;//false表示存储量不足
//int mincost;//最小花费
//int maxflow;//最大流
pair<int,int>pr; int totalcost;
int totalflow;
int S,T;//超级源点,超级汇点 while(~scanf("%d%d%d",&N,&M,&K)){
if(N==&&M==&&K==)break;
memset(totalneed,,sizeof(totalneed));
memset(totalstorage,,sizeof(totalstorage));
flag=true;
totalcost=;
totalflow=; for(int i=;i<N;++i){
for(int j=;j<K;++j){
scanf("%d",&need[i][j]);
totalneed[j]+=need[i][j];
}
}
for(int i=;i<M;++i){
for(int j=;j<K;++j){
scanf("%d",&storage[i][j]);
totalstorage[j]+=storage[i][j];
}
}
for(int i=;i<K;++i){
if(totalneed[i]>totalstorage[i]){
flag=false;
break;
}
} for(int k=;k<K;++k){
solve.init();//初始化要放这里
for(int i=;i<N;++i){
for(int j=;j<M;++j){
scanf("%d",&cost[i][j]);
solve.addedge(j,M+i,INF,cost[i][j]);//注意加边:j->M+i
}
}
if(flag==false)continue; //超级源点
S=N+M;
for(int v=;v<M;++v){
solve.addedge(S,v,storage[v][k],);
}
//超级汇点
T=N+M+;
for(int v=;v<N;++v){
solve.addedge(M+v,T,need[v][k],);
} pr=solve.mincostmaxflow(S,T,N+M+);
if(pr.first<totalneed[k])
flag=false; totalcost+=pr.first;
totalflow+=pr.second;
} if(flag==false)printf("-1\n");
else printf("%d\n",totalcost);
}
return ;
}
POJ - 2516 Minimum Cost(最小费用最大流)的更多相关文章
- POJ 2516 Minimum Cost [最小费用最大流]
题意略: 思路: 这题比较坑的地方是把每种货物单独建图分开算就ok了. #include<stdio.h> #include<queue> #define MAXN 500 # ...
- Poj 2516 Minimum Cost (最小花费最大流)
题目链接: Poj 2516 Minimum Cost 题目描述: 有n个商店,m个仓储,每个商店和仓库都有k种货物.嘛!现在n个商店要开始向m个仓库发出订单了,订单信息为当前商店对每种货物的需求 ...
- POJ2516:Minimum Cost(最小费用最大流)
Minimum Cost Time Limit: 4000MS Memory Limit: 65536K Total Submissions: 19088 Accepted: 6740 题目链 ...
- POJ2516 Minimum Cost —— 最小费用最大流
题目链接:https://vjudge.net/problem/POJ-2516 Minimum Cost Time Limit: 4000MS Memory Limit: 65536K Tota ...
- Minimum Cost(最小费用最大流)
Description Dearboy, a goods victualer, now comes to a big problem, and he needs your help. In his s ...
- POJ 2516 Minimum Cost (最小费用最大流)
POJ 2516 Minimum Cost 链接:http://poj.org/problem?id=2516 题意:有M个仓库.N个商人.K种物品.先输入N,M.K.然后输入N行K个数,每一行代表一 ...
- POJ 2516 Minimum Cost (网络流,最小费用流)
POJ 2516 Minimum Cost (网络流,最小费用流) Description Dearboy, a goods victualer, now comes to a big problem ...
- POJ 2516 Minimum Cost(最小费用流)
Description Dearboy, a goods victualer, now comes to a big problem, and he needs your help. In his s ...
- POJ 2516 Minimum Cost
每个物品分开做最小费用最大流. #include<cstdio> #include<cstring> #include<cmath> #include<vec ...
随机推荐
- BootStrap学习02栅格系统
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- Python+selenium登录测试
我们以登录新浪微博为案例来讲解,首先进入登录页面,输入用户名和密码,点击登录按钮,并且获得用户信息以验证是否登录成功. Web地址:https://login.sina.com.cn/signup/s ...
- PHP建立和删除目录
<?php/*linux中的文件权限filedir 用户 组 其它 rwx rwx rwx 读写执行 6 4 6 读写 读 读写 7 7 7 rw_ r__ rw_ r__ _w_ ___ r ...
- Android资源目录结构
资源目录结构 res为资源目录,主要以xml语法编写静态的资源. 资源的命名标准:小写字母和数字,且以小写字母开头. 资源的生成,为了和java语法沟通,资源文件会自动的生成在[gen]目录的R.ja ...
- Permutations(排列问题,DFS回溯)
Given a collection of numbers, return all possible permutations. For example,[1,2,3] have the follow ...
- uva 1364
刘书上例题 #include <cstdio> #include <cstdlib> #include <cmath> #include <set> # ...
- codeforces 873E(枚举+rmq)
题意 有n(n<=3000)个人参与acm比赛,每个人都有一个解题数,现在要决定拿金牌的人数cnt1,拿银牌的人数cnt2,拿铜牌的人数cnt3,各自对应一个解题数区间[d1,c1],[d2,c ...
- SHELL脚本实现分区
写一个脚本(前提:请为虚拟机新增一块硬盘,架设它为/dev/sdb),为指定的硬盘创建分区 1,列出当前系统上所有的磁盘,让用户选择,如果选择quit则退出脚本:如果用户选择错误,就让用户重新选择 2 ...
- TeamCity - Docker创建
// 创建Server docker run -it --name teamcity-server-instance \-v /home/tc_datadir:/data/teamcity_serve ...
- 基于Hexo + Git + Nginx的博客发布
进过上一篇<树莓派搭建私人服务器>,我们已经有一个私人服务器了,现在需要做点什么实际事情了,先搭一个博客分享自己的经验吧. 相关文章:1.<树莓派搭建私人服务器>(http:/ ...