题目链接

题意

有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. 2.golang应用目录结构和GOPATH概念

    golang 语言有一个GOPATH的概念就是当前工作目录 [root@localhost golang_test]# tree . ├── bin │   └── hello ├── first.g ...

  2. 动画可以暂停animation-play-state

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  3. HDU-1269 迷宫城堡(连通分量)

    迷宫城堡 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submi ...

  4. 【JAVA】格式化打印printf的使用

    格式化打印printf的使用 import java.util.Date; /** * 使用printf输出 */ /**关键技术点 * 使用java.io.PrintStream的printf方法实 ...

  5. C# 给DataTable去重

    using System; using System.Data; namespace DelegateTest { public class Program { public static void ...

  6. Sql Server 压缩数据库占用空间

    1.删除数据库库中不必要的数据2. 在数据库上右击,任务,收缩,文件,在收缩操作上选择在未使用的空间前重新组织页,将文件收缩到的最后一行为最小为XXM,在前面的输入框中填入该值,然后点击确定3.分离该 ...

  7. gay绿论第一章两点注意事项

    1.不可能事件的概率为0,但概率为0的事件不一定是不可能事件,例如从自然数中取一个数结果是1的概率,从极限角度看,分子是1,分母是∞,结果是0,但它显然是有可能发生的,所以不是不可能事件. 2.两事件 ...

  8. Windows中的Work线程和GUI线程

    Windows线程分为两种:Worker线程.GUI线程 worker线程:是指完全不牵扯到图形用户界面(GUI),纯粹做运算的线程. GUI线程:负责建造窗口以及处理消息循环(拥有消息队列).任何一 ...

  9. Python---基础---dict和set

    2019-05-20 ------------------------------ 写一个程序来管理用于登陆系统的用户信息:登录名字和密码,登录用户账号建立后,已存在用户可以用登陆名字和密码重返系统, ...

  10. Android SDK说明(图)