BZOJ_1070_[SCOI2007]修车_费用流
Description
同一时刻有N位车主带着他们的爱车来到了汽车维修中心。维修中心共有M位技术人员,不同的技术人员对不同
的车进行维修所用的时间是不同的。现在需要安排这M位技术人员所维修的车及顺序,使得顾客平均等待的时间最
小。 说明:顾客的等待时间是指从他把车送至维修中心到维修完毕所用的时间。
Input
第一行有两个m,n,表示技术人员数与顾客数。 接下来n行,每行m个整数。第i+1行第j个数表示第j位技术人
员维修第i辆车需要用的时间T。
Output
最小平均等待时间,答案精确到小数点后2位。
Sample Input
2 2
3 2
1 4
3 2
1 4
Sample Output
1.50
HINT
数据范围: (2<=M<=9,1<=N<=60), (1<=T<=1000)
对于1个技术人员p,假设他第i个修的车是第j辆车,对等待时间的贡献为(k-i)*t[j][p]其中k为安排这个人修车的数量。
考虑把每个技术人员拆成n个点,对于每辆车i,向这些点连一条容量1费用为时间k*t[i][j],表示第j个人倒数第k个时间段修第i辆车。然后源点向每个车连容1费0,每个拆点后的技术人员向汇点连容1费0,然后跑最小费用最大流即可。
代码:
#include <stdio.h>
#include <string.h>
#include <algorithm>
using namespace std;
#define N 850
#define M 250050
#define S (n*m+m+1)
#define T (n*m+m+2)
#define inf 0x3f3f3f3f
int head[N],to[M],nxt[M],flow[M],val[M],cnt=1,n,m,inq[N];
int t[70][70],dis[N],Q[N],l,r,path[N];
inline void add(int u,int v,int f,int w) {
to[++cnt]=v; nxt[cnt]=head[u]; head[u]=cnt; flow[cnt]=f; val[cnt]=w;
to[++cnt]=u; nxt[cnt]=head[v]; head[v]=cnt; flow[cnt]=0; val[cnt]=-w;
}
bool spfa() {
memset(dis,0x3f,sizeof(dis));
memset(path,0,sizeof(path));
dis[S]=0; l=r=0; Q[r++]=S; inq[S]=1;
int i;
while(l!=r) {
int x=Q[l++];if(l==n*m+m+1) l=0; inq[x]=0;
for(i=head[x];i;i=nxt[i]) if(flow[i]&&dis[to[i]]>dis[x]+val[i]) {
dis[to[i]]=dis[x]+val[i];
path[to[i]]=i^1;
if(!inq[to[i]]) {
inq[to[i]]=1; Q[r++]=to[i]; if(r==n*m+m+1) r=0;
}
}
}
return dis[T]<inf;
}
void mcmf() {
int minc=0,maxf=0,i;
while(spfa()) {
int nf=1<<30;
for(i=T;i!=S;i=to[path[i]]) {
nf=min(nf,flow[path[i]^1]);
}
for(i=T;i!=S;i=to[path[i]]) {
flow[path[i]]+=nf;
flow[path[i]^1]-=nf;
minc+=nf*val[path[i]^1];
}
maxf+=nf;
}
printf("%.2lf\n",1.0*minc/m);
}
int main() {
scanf("%d%d",&n,&m);
int i,j,k;
for(i=1;i<=m;i++) {
for(j=1;j<=n;j++) {
scanf("%d",&t[i][j]);
}
}
for(i=1;i<=m;i++) {
add(S,i,1,0);
for(j=1;j<=n;j++) {
for(k=1;k<=m;k++) {
add(i,j*m+k,1,k*t[i][j]);
}
}
}
for(i=1;i<=n;i++) {
for(j=1;j<=m;j++) {
add(i*m+j,T,1,0);
}
}
mcmf();
}