bzoj1070题解

【解题思路】

  考虑拆点,得到一个二分图:左边点<i,j>表示第i个技师按顺序第j辆修的车,右边点k表示第k个车主,连接左右的边表示第k个车主可能成为第i个技师的第j个客户。

  因为是二分图,所以直接跑KM即可,复杂度O(n3m);或者考虑费用流,左图都和源点连边,右图都和汇点连边,容费均为1,跑个流即可,复杂度O(松)。

【参考代码】

  费用流实现:

 1 #pragma GCC optimize(2)
 2 #include <cstdio>
 3 #include <cstring>
 4 #define REP(i,low,high) for(register int i=(low);i<=(high);i++)
 5 #define INF 0x7f7f7f7f
 6 using namespace std;
 7 template<typename T> inline bool getmin(T &tar,const T &pat) {return pat<tar?tar=pat,1:0;}
 8 const static int T=1001; static int E=1; bool inq[T+10]; int w,c,hed[T+10],pre[T+10],que[T<<3]; long long dst[T+10],tim[10][100];
 9 struct edge
10 {
11     int fr,to,fl,nx; long long vl;
12     edge() {fl=INF;}
13     edge(const int &x,const int &y,const int &f,const long long &v) {fr=x,to=y,fl=f,vl=v,nx=hed[x],hed[x]=E;}
14 }edg[100010];
15 inline void add_edge(const int &fr,const int &to,const int &fl,const long long &vl) {edg[++E]=edge(fr,to,fl,vl),edg[++E]=edge(to,fr,0,-vl);}
16 inline bool SPFA()
17 {
18     memset(dst,0x7f,sizeof dst),dst[0]=que[0]=0ll,memset(inq,0,sizeof inq),inq[0]=1;
19     for(int head=-1,tail=0;head++<tail;)
20     {
21         int now=que[head];
22         for(int i=hed[now];i;i=edg[i].nx)
23         {
24             int p=edg[i].to; if(edg[i].fl&&getmin(dst[p],dst[now]+edg[i].vl)) {pre[p]=i; if(!inq[p]) inq[que[++tail]=p]=1;}
25         }
26         inq[now]=0;
27     }
28     return dst[T]<INF;
29 }
30 int main()
31 {
32     scanf("%d%d",&w,&c),memset(hed,0,sizeof hed); REP(i,1,c) REP(j,1,w) scanf("%lld",&tim[j][i]); int n=w*c; long long ans=0ll;
33     REP(i,1,n) add_edge(0,i,1,0ll); REP(i,1,c) add_edge(n+i,T,1,0ll); REP(i,1,w) REP(j,1,c) REP(k,1,c) add_edge((i-1)*c+j,n+k,1,tim[i][k]*j);
34     while(SPFA())
35     {
36         int mnr=INF; for(int i=pre[T];i;i=pre[edg[i].fr]) getmin(mnr,edg[i].fl);
37         for(int i=pre[T];i;i=pre[edg[i].fr]) edg[i].fl-=mnr,edg[i^1].fl+=mnr,ans+=edg[i].vl*mnr;
38     }
39     return printf("%.2lf\n",(double)ans/c),0;
40 }
View Code

 

转载于:https://www.cnblogs.com/spactim/p/6623446.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值