2879. [NOI2012]美食节【费用流】
Description
CZ市为了欢迎全国各地的同学,特地举办了一场盛大的美食节。作为一个喜欢尝鲜的美食客,小M自然不愿意错过这场盛宴。他很快就尝遍了美食节所有的美食。然而,尝鲜的欲望是难以满足的。尽管所有的菜品都很可口,厨师做菜的速度也很快,小M仍然觉得自己桌上没有已经摆在别人餐桌上的美食是一件无法忍受的事情。于是小M开始研究起了做菜顺序的问题,即安排一个做菜的顺序使得同学们的等待时间最短。小M发现,美食节共有n种不同的菜品。每次点餐,每个同学可以选择其中的一个菜品。总共有m个厨师来制作这些菜品。当所有的同学点餐结束后,菜品的制作任务就会分配给每个厨师。然后每个厨师就会同时开始做菜。厨师们会按照要求的顺序进行制作,并且每次只能制作一人份。此外,小M还发现了另一件有意思的事情: 虽然这m个厨师都会制作全部的n种菜品,但对于同一菜品,不同厨师的制作时间未必相同。他将菜品用1, 2, ..., n依次编号,厨师用1, 2, ..., m依次编号,将第j个厨师制作第i种菜品的时间记为 ti,j 。小M认为:每个同学的等待时间为所有厨师开始做菜起,到自己那份菜品完成为止的时间总长度。换句话说,如果一个同学点的菜是某个厨师做的第k道菜,则他的等待时间就是这个厨师制作前k道菜的时间之和。而总等待时间为所有同学的等待时间之和。现在,小M找到了所有同学的点菜信息: 有 pi 个同学点了第i种菜品(i=1, 2, ..., n)。他想知道的是最小的总等待时间是多少。
Input
输入文件的第1行包含两个正整数n和m,表示菜品的种数和厨师的数量。
第2行包含n个正整数,其中第i个数为pi,表示点第i种菜品的人数。
接下来有n行,每行包含m个非负整数,这n行中的第i行的第j个数为ti,j,表示第j个厨师制作第i种菜品所需的时间。
输入文件中每行相邻的两个数之间均由一个空格隔开,行末均没有多余空格。
Output
输出仅一行包含一个整数,为总等待时间的最小值。
Sample Input
3 1 1
5 7
3 6
8 9
Sample Output
47
【样例说明】
厨师1先制作1份菜品2,再制作2份菜品1。点这3道菜的3个同学的等待时间分别为3,3+5=8,3+5+5=13。
厨师2先制作1份菜品1,再制作1份菜品3。点这2道菜的2个同学的等待时间分别为7,7+9=16。
总等待时间为3+8+13+7+16=47。
虽然菜品1和菜品3由厨师1制作更快,如果这些菜品都由厨师1制作,总等待时间反而更长。如果按上述的做法,将1份菜品1和1份菜品3调整到厨师2制作,这样厨师2不会闲着,总等待时间更短。
可以证明,没有更优的点餐方案。
【数据规模及约定】
对于100%的数据,n <= 40, m <= 100, p <= 800, ti,j <= 1000(其中p = ∑pi,即点菜同学的总人数)。
每组数据的n、m和p值如下:
测试点编号 n m p 
1 n = 5 m = 5 p = 10 
2 n = 40 m = 1 p = 400 
3 n = 40 m = 2 p = 300 
4 n = 40 m = 40 p = 40 
5 n = 5 m = 40 p = 100 
6 n = 10 m = 50 p = 200 
7 n = 20 m = 60 p = 400 
8 n = 40 m = 80 p = 600 
9 n = 40 m = 100 p = 800 
10 n = 40 m = 100 p = 800
这题和修车那个题一毛一样……
只不过这个数据范围恶心的一比……
普通做法肯定不行
所以我用了(从题解里抄的)动态加边
每次找增广路径的时候,若当前是某个厨师的倒数第k道菜,且该厨师的边只连到倒数第k道
那么就把该厨师倒数第k+1道连上边即可
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<queue>
#define MAXN (100000+10)
#define MAXM (5000000+10)
using namespace std;
queue<int>q;
bool visit[MAXN];
int pre[MAXN];
int n,m,k,s,e=,Ans,Fee;
int num_edge;
int head[MAXN];
int dis[MAXN];
bool used[MAXN];
int INF;
int t[][],p[],sump;
int cnt[MAXN];
struct node
{
int to;
int next;
int Flow;//残留网络
int Cost;
} edge[MAXM*]; void add(int u,int v,int l,int c)
{
edge[++num_edge].to=v;
edge[num_edge].next=head[u];
edge[num_edge].Flow=l;
edge[num_edge].Cost=c;
head[u]=num_edge;
} bool Spfa(int s,int e)
{
// memset(pre,-1,sizeof(pre));
memset(dis,0x7f,sizeof(dis));
q.push(s);
dis[s]=;
used[s]=true;
while (!q.empty())
{
int x=q.front();
q.pop();
for (int i=head[x]; i!=; i=edge[i].next)
if (dis[x]+edge[i].Cost<dis[edge[i].to] && edge[i].Flow>)
{
dis[edge[i].to]=edge[i].Cost+dis[x];
pre[edge[i].to]=i;
if (!used[edge[i].to])
{
used[edge[i].to]=true;
q.push(edge[i].to);
}
}
used[x]=false;
}
return (dis[e]!=INF);
} void MCMF(int s,int e)
{
while (Spfa(s,e))
{
int d=INF,x;
for (int i=e; i!=s; i=edge[((pre[i]-)^)+].to)
d=min(d,edge[pre[i]].Flow);
for (int i=e; i!=s; i=edge[((pre[i]-)^)+].to)
{
edge[pre[i]].Flow-=d;
edge[((pre[i]-)^)+].Flow+=d;
if (edge[((pre[i]-)^)+].to==s) x=i;//这一次增广的是厨师那坨点的哪个点
}
Fee+=d*dis[e]; int a=(x-)/sump+,b=(x-)%sump+;
if (b<cnt[a]) return;
cnt[a]++;
for (int i=; i<=n; ++i)
{
add((a-)*sump+b+,m*sump+i,,cnt[a]*t[i][a]);
add(m*sump+i,(a-)*sump+b+,,-cnt[a]*t[i][a]);
}
}
} int main()
{
memset(&INF,0x7f,sizeof(INF));
scanf("%d%d",&n,&m);
for (int i=; i<=m; ++i) cnt[i]=;
for (int i=; i<=n; ++i)
scanf("%d",&p[i]),sump+=p[i];
for (int i=; i<=n; ++i)
for (int j=; j<=m; ++j)
scanf("%d",&t[i][j]);
for (int i=; i<=m*sump; ++i)
{
add(s,i,,);
add(i,s,,);
}
for (int i=; i<=n; ++i)
{
add(m*sump+i,e,p[i],);
add(e,m*sump+i,,);
}
for (int i=; i<=m; ++i) //第i个厨师
for (int k=; k<=n; ++k)
{
add((i-)*sump+,m*sump+k,,t[k][i]);
add(m*sump+k,(i-)*sump+,,-t[k][i]);
}
MCMF(s,e);
printf("%d\n",Fee);
}
2879. [NOI2012]美食节【费用流】的更多相关文章
- BZOJ 2879: [Noi2012]美食节( 费用流 + 动态加边 )
		
倒着做菜..然后考虑为当前的人做菜对后面的人的影响就可以了..要动态加边 --------------------------------------------------------------- ...
 - BZOJ.2879.[NOI2012]美食节(费用流SPFA)
		
题目链接 /* 同"修车":对于每个厨师拆成p个点表示p个时间点,每个人向m个厨师每个时间点连边 这样边数O(nmp)+网络流 ≈O(nm*p^2)(假设SPFA线性) = GG ...
 - BZOJ 2879 [Noi2012]美食节 | 费用流 动态开点
		
这道题就是"修车"的数据加强版--但是数据范围扩大了好多,应对方法是"动态开点". 首先先把"所有厨师做的倒数第一道菜"和所有菜连边,然后跑 ...
 - [NOI2012]美食节(费用流)
		
题目描述 CZ市为了欢迎全国各地的同学,特地举办了一场盛大的美食节.作为一个喜欢尝鲜的美食客,小M自然不愿意错过这场盛宴.他很快就尝遍了美食节所有的美食.然而,尝鲜的欲望是难以满足的.尽管所有的菜品都 ...
 - [NOI2012]美食节——费用流(带权二分图匹配)+动态加边
		
题目描述 小M发现,美食节共有n种不同的菜品.每次点餐,每个同学可以选择其中的一个菜品.总共有m个厨师来制作这些菜品.当所有的同学点餐结束后,菜品的制作任务就会分配给每个厨师.然后每个厨师就会同时开始 ...
 - 【bzoj2879】[Noi2012]美食节  费用流+动态加边
		
原文地址:http://www.cnblogs.com/GXZlegend 题目描述 CZ市为了欢迎全国各地的同学,特地举办了一场盛大的美食节.作为一个喜欢尝鲜的美食客,小M自然不愿意错过这场盛宴.他 ...
 - 【BZOJ 2879】[Noi2012]美食节 费用流
		
思路同修车,就是多了一个骚气的操作:动态加边,我们通过spfa流的过程可以知道,我们一次只会跑一流量,最后一层边跑过就不会再悔改,所以说我们只会用到一大片里面的很少的点,所以我们如果可以动态加边的话我 ...
 - [BZOJ2879][NOI2012]美食节(费用流)
		
设sm为所有p之和,套路地对每道菜建一个点,将每个厨师拆成sm个点,做的倒数第i道菜的代价为time*i. S向每道菜连边<0,p[i]>(前者为代价后者为流量),i菜到j厨师的第k个点连 ...
 - [BZOJ2879] [Noi2012] 美食节 (费用流 & 动态加边)
		
Description CZ市为了欢迎全国各地的同学,特地举办了一场盛大的美食节.作为一个喜欢尝鲜的美食客,小M自然不愿意错过这场盛宴.他很快就尝遍了美食节所有的美食.然而,尝鲜的欲望是难以满足的.尽 ...
 - BZOJ 2879: [Noi2012]美食节
		
2879: [Noi2012]美食节 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 1834 Solved: 969[Submit][Status] ...
 
随机推荐
- ASP.NET MVC控制器里捕获视图的错误验证信息(ErrorMessage)
			
ViewModel类: /// <summary> /// 评论用验证视图 /// </summary> public partial class VCreateShopCom ...
 - Spring系列之——springboot解析resources.application.properties文件
			
摘要:本文通过讲解如何解析application.properties属性,介绍了几个注解的运用@Value @ConfigurationProperties @EnableConfiguration ...
 - Java 使用new Thread和线程池的区别
			
本文转至:https://www.cnblogs.com/cnmenglang/p/6273761.html , 孟凡柱的专栏 的博客,在此谢谢博主! 1.new Thread的弊端执行一个异步任务你 ...
 - 使用rem编写自适应屏幕网页造成div被span撑高的解决办法
			
原始代码: <html> <head> <meta charset="utf-8"> <meta content="ie=edg ...
 - windows多线程窗口程序设计
			
掌握windows基于消息驱动的窗口应用程序设计的基本方法,掌握窗口程序资源的概念与设计,掌握常用的消息的程序处理方法,掌握文字图形输出相关函数编程.掌握设计的基本方法(选项),掌握时钟消息设计动画程 ...
 - Java新建线程的两种方式
			
Java新建线程有两种方式,一种是通过继承Thread类,一种是实现Runnable接口,下面是新建线程的两种方式. 我们假设有个竞赛,有一个选手A做俯卧撑,一个选手B做仰卧起坐.分别为两个线程: p ...
 - MySQL 中文未正常显示
			
关于MySQL中文乱码问题 最近发现,在MySQL的dos客户端输出窗口中查询表中的数据时,表中的中文数据都显示成乱码: 之所以会显示乱码,就是因为MySQL客户端输出窗口显示中文时使用的字符编码不对 ...
 - [h5+api]移动app开发用到的微信好友,朋友圈,qq好友,新浪微博分享合集
			
适用H5+环境,能够使用plus方法的移动app中 /** * Created by HBuilder. * User: tyx * Date: 2018-11-21 * Time: 17:28:51 ...
 - bzoj P4870: [Shoi2017]组合数问题——solution
			
题意:求解—— $$(C^{r}_{nk}+C^{r+k}_{nk}+C^{r+2k}_{nk}+...+C^{r+(n-1)k}_{nk}+...)mod(P)$$ 其中$C^{m}_{n}$表示从 ...
 - CSS属性之border
			
css的border属性相信大家都不陌生了,就是给元素加个边框嘛,在不同的盒模型下,会给元素的宽高带来怎样的影响,相信大家也都很熟悉了,这里就不再赘述,只说说大家平时没有怎么留意的东西. 0.bord ...