题目描述
同一时刻有N位车主带着他们的爱车来到了汽车维修中心。维修中心共有M位技术人员,不同的技术人员对不同的车进行维修所用的时间是不同的。现在需要安排这M位技术人员所维修的车及顺序,使得顾客平均等待的时间最小。
说明:顾客的等待时间是指从他把车送至维修中心到维修完毕所用的时间。
输入格式
第一行有两个数M,N,表示技术人员数与顾客数。
接下来n行,每行m个整数。第i+1行第j个数表示第j位技术人员维修第i辆车需要用的时间T。
输出格式
最小平均等待时间,答案精确到小数点后2位。
题解:
将m个人拆成n个点,表示这个人倒数是第几个修车的,比如第k辆车连向第i个人的第j个点,表示第i个人是倒数第j个修车的,修的是第k辆车,这样人们的等待时间就可以算出,因为这辆车是倒数第j个修的,那么一共就有jl辆车会等待a[i][k]的时间。
这样就很简单了,按照源点->车->人的拆点->汇点建图,EK一套就OK了
AC代码:
#pragma GCC optimize(2)
#include<bits/stdc++.h>
#include<ext/rope>
using namespace std;
using namespace __gnu_cxx;
#define LL long long
#define pii pair<int,int>
#define mp(a,b) make_pair(a,b)
const int MAXN = 1e5+50;
const int MAXM = 2e6+50;
const int MOD = 1e9+7;
const int INF = 0x3f3f3f3f;
int m,n,s,t,tot=1,head[MAXN],to[MAXM],w[MAXM],co[MAXM],nxt[MAXM];
int a[65][65],dis[MAXN],vis[MAXN],flow[MAXN],pre[MAXN];
inline void ade(int u,int v,int ww,int cost){
to[++tot]=v; w[tot]=ww; co[tot]=cost; nxt[tot]=head[u]; head[u]=tot;
}
inline void add(int u,int v,int w,int cost){ ade(u,v,w,cost); ade(v,u,0,-cost); }
inline int spfa(){
for(int i=0;i<=t;i++) dis[i]=INF;
queue<int> que; que.push(s); dis[s]=0; vis[s]=1; flow[s]=INF;
while(!que.empty()){
int u=que.front(); que.pop(); vis[u]=0;
for(int i=head[u];i;i=nxt[i]){
if(w[i] && dis[to[i]]>dis[u]+co[i]){
dis[to[i]]=dis[u]+co[i]; pre[to[i]]=i;
flow[to[i]]=min(flow[u],w[i]);
if(!vis[to[i]]) vis[to[i]]=1,que.push(to[i]);
}
}
}
return dis[t]!=INF;
}
inline double EK(){
double res=0;
while(spfa()){
res+=dis[t]*flow[t]; int x=t;
while(x!=s){
int i=pre[x];
w[i]-=flow[t]; w[i^1]+=flow[t]; x=to[i^1];
}
}
return res;
}
signed main(){
#ifndef ONLINE_JUDGE
freopen("C:\\Users\\Administrator\\Desktop\\in.txt","r",stdin);
#endif // ONLINE_JUDGE
scanf("%d%d",&m,&n); s=0; t=n*m+n+1;
for(int i=1;i<=n;i++) add(s,i,1,0);
for(int i=1;i<=n;i++) for(int j=1;j<=m;j++) scanf("%d",&a[i][j]);
for(int i=1;i<=m;i++) for(int j=1;j<=n;j++) add(n+(i-1)*n+j,t,1,0);
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
for(int k=1;k<=n;k++)
add(i,n+(j-1)*n+k,1,k*a[i][j]);
printf("%.2f\n",EK()/n);
return 0;
}