题目链接

题意

有N个商家它们需要货物源,还有M个货物供应商,N个商家需要K种物品,每种物品都有对应的需求量,M个商家每种物品都是对应的存货,然后再是K个N*M的矩阵表示了K个物品从供货商运送到商家的单位上的价钱,那么就是标准的最大流最小费用了,我们只需要建立这样的边,对于所有的供应商都与源点建立流的大小为拥有的个数的边、与商家建立无穷大的边并且边的代价是单位流的代价,然后再由商家出发到达汇点建立流大小为其需要的边,与汇点和源点建立的边的代价都是0。

思路

一开始的时候,我计划直接跑一次费用流,但是这样跑了之后,发现了会T,然后考虑到有N个需求商,还有M个提供商,如果想直接一遍跑完的话,点的个数是(N + M + N * K + M * K)这样子点的个数就太多了,但是我们可以换一下,如果分成K次来讨论的话,是不是可以优化下来时间复杂度。

所以,我们分成K次,每次处理对应的货物种类,然后建边分别跑费用流,具体如下。

#include <iostream>
#include <cstdio>
#include <cmath>
#include <string>
#include <cstring>
#include <algorithm>
#include <limits>
#include <vector>
#include <stack>
#include <queue>
#include <set>
#include <map>
#define lowbit(x) ( x&(-x) )
#define pi 3.141592653589793
#define e 2.718281828459045
#define INF 0x3f3f3f3f
#define HalF (l + r)>>1
#define lsn rt<<1
#define rsn rt<<1|1
#define Lson lsn, l, mid
#define Rson rsn, mid+1, r
#define QL Lson, ql, qr
#define QR Rson, ql, qr
#define myself rt, l, r
using namespace std;
const int maxN = , S = ;
int N, M, K, T, need[], have[], shop[][], good[][], r[maxN][maxN], c[maxN][maxN];
int pre[maxN], dist[maxN], Flow[maxN], ans;
queue<int> Q;
bool inque[maxN];
bool spfa()
{
memset(pre, , sizeof(pre)); memset(dist, INF, sizeof(dist)); memset(inque, false, sizeof(inque));
Q.push(S); inque[S] = true; dist[S] = ; Flow[S] = INF;
while(!Q.empty())
{
int u = Q.front(); inque[u] = false; Q.pop();
for(int i=; i<=T; i++)
{
if(r[u][i] && dist[i] > dist[u] + c[u][i])
{
dist[i] = dist[u] + c[u][i];
Flow[i] = min(Flow[u], r[u][i]);
pre[i] = u;
if(!inque[i])
{
inque[i] = true;
Q.push(i);
}
}
}
}
return pre[T];
}
int EK()
{
int ans = ;
while(spfa())
{
int now = T, las = pre[now];
while(now)
{
r[las][now] -= Flow[T];
r[now][las] += Flow[T];
now = las;
las = pre[now];
}
ans += Flow[T] * dist[T];
}
return ans;
}
inline void init()
{
ans = ; T = N + M + ;
memset(need, , sizeof(need));
memset(have, , sizeof(have));
}
int main()
{
while(scanf("%d%d%d", &N, &M, &K) && (N || M || K))
{
init();
for(int i=; i<=N; i++)
{
for(int j=; j<=K; j++)
{
scanf("%d", &shop[i][j]);
need[j] += shop[i][j];
}
}
for(int i=; i<=M; i++)
{
for(int j=; j<=K; j++)
{
scanf("%d", &good[i][j]);
have[j] += good[i][j];
}
}
bool flag = true;
for(int i=; i<=K; i++)
{
if(need[i] > have[i])
{
flag = false;
break;
}
}
for(int i=; i<=K; i++)
{
memset(r, , sizeof(r));
memset(c, , sizeof(c));
for(int j=; j<=N; j++)
{
r[S][j] = shop[j][i];
for(int kk=; kk<=M; kk++)
{
scanf("%d", &c[j][N + kk]);
c[N + kk][j] = -c[j][N + kk];
r[j][N + kk] = INF;
}
}
if(!flag) continue;
for(int j=; j<=M; j++) r[N + j][T] = good[j][i];
ans += EK();
}
if(!flag) { printf("-1\n"); continue; }
printf("%d\n", ans);
}
return ;
}

Minimum Cost 【POJ - 2516】【网络流最小费用最大流】的更多相关文章

  1. kuangbin专题专题十一 网络流 Minimum Cost POJ - 2516

    题目链接:https://vjudge.net/problem/POJ-2516 思路:对于每种商品跑最小费用最大流,如果所有商品和人一起建图跑,O(v^2*m)数量级太大,会超时. 把店里的商品拆点 ...

  2. 网络流(最小费用最大流):POJ 2135 Farm Tour

    Farm Tour Time Limit: 1000ms Memory Limit: 65536KB This problem will be judged on PKU. Original ID: ...

  3. Minimum Cost POJ - 2516(模板题。。没啥好说的。。)

    题意: 从发货地到商家 送货 求送货花费的最小费用... 有m个发货地,,,n个商家,,每个商家所需要的物品和物品的个数都不一样,,,每个发货地有的物品和物品的个数也不一样,,, 从不同的发货地到不同 ...

  4. HDU 6118 度度熊的交易计划(网络流-最小费用最大流)

    度度熊参与了喵哈哈村的商业大会,但是这次商业大会遇到了一个难题: 喵哈哈村以及周围的村庄可以看做是一共由n个片区,m条公路组成的地区. 由于生产能力的区别,第i个片区能够花费a[i]元生产1个商品,但 ...

  5. POJ 3680 Intervals 最小费用最大流(MCMF算法)

    题意:给出 n ,k 表示接下来给你 n 段开区间,每段区间都有它的权值,问选出一些区间,使它的权值最大,并且在实轴上的每个点,不得超过 k次被覆盖. 思路:首先要理解建图思路,首先有一个基图,相邻点 ...

  6. Minimum Cost POJ - 2516 (模板题 spfa最小费用最大流)

    题意: 人回家,一步一块钱,有x个人,y个房子,求能回家的最大人数且使之费用最小 解析: 就是....套模板,,,, 建图(⊙﹏⊙)...要仔细观察呐 对于人拆不拆都可以  都能过,,,,这里贴上拆开 ...

  7. E - Minimum Cost - POJ 2516(最小费)

    题目大意:N个客户,M个供货商,K种商品,现在知道每个客户对每种商品的需求量,也知道每个供货商每种商品的持有量,和供货商把一种商品运送到每个客户的单位花费.现在想知道如果能满足所有客户的最小花费是多少 ...

  8. POJ-2516-Minimum Cost(网络流, 最小费用最大流)

    链接: https://vjudge.net/problem/POJ-2516 题意: Dearboy, a goods victualer, now comes to a big problem, ...

  9. 网络流--最小费用最大流MCMF模板

    标准大白书式模板 #include<stdio.h> //大概这么多头文件昂 #include<string.h> #include<vector> #includ ...

随机推荐

  1. 【洛谷p1314】聪明的质监员

    聪明的质监员[题目链接] 有关算法: 二分答案: 但是你只二分答案是不够的,因为你check会炸,所以还要考虑前缀和: 首先假装我们的check已经写好了,main函数: int main() { n ...

  2. Python学习-第三天-面向对象编程基础

    Python学习-第三天-面向对象编程基础 类和对象 简单的说,类是对象的蓝图和模板,而对象是类的实例.这个解释虽然有点像用概念在解释概念,但是从这句话我们至少可以看出,类是抽象的概念,而对象是具体的 ...

  3. HDU-1754 I Hate It(线段树,区间最大值)

    很多学校流行一种比较的习惯.老师们很喜欢询问,从某某到某某当中,分数最高的是多少.  这让很多学生很反感. 不管你喜不喜欢,现在需要你做的是,就是按照老师的要求,写一个程序,模拟老师的询问.当然,老师 ...

  4. JS—图片压缩上传(单张)

    *vue+webpack环境,这里的that指到vue实例 <input type="file" name="file" accept="ima ...

  5. POJ3321[苹果树] 树状数组/线段树 + dfs序

    Apple Tree Time Limit: 2000MS   Memory Limit: 65536K Total Submissions:39452   Accepted: 11694 Descr ...

  6. Add JWT Bearer Authorization to Swagger and ASP.NET Core

    Add JWT Bearer Authorization to Swagger and ASP.NET Core     If you have an ASP.NET Core web applica ...

  7. 帝国CMS 调用专题[eshowzt]标签改为灵动标签[e:loop]的方法

    1.eshowzt说明及示例 [eshowzt]标签模板ID,专题类别ID,显示专题数[/eshowzt] 示例需求:将推荐专题后,显示9个指定专题 原来写法: <span>[eshowz ...

  8. websocket在springboot+vue中的使用

    1.websocket在springboot中的一种实现 在java后台中,websocket是作为一种服务端配置,其配置如下 @Configuration public class WebSocke ...

  9. linux man命令后面各种括号的意义

    圆括号 我们经常会看到 在说一个对象的man page 的时候,会有这样的格式: mmap(2) shm_open(3) 这个后面的数字是什么意思呢,通过 man man 命令就可以知道,这个是数字是 ...

  10. java类实现序列化的方法?collection框架中实现什么样的接口