zoukankan      html  css  js  c++  java
  • BZOJ 1070: [SCOI2007]修车

    1070: [SCOI2007]修车

    Time Limit: 1 Sec  Memory Limit: 128 MB
    Submit: 4983  Solved: 2049
    [Submit][Status][Discuss]

    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

    Sample Output

    1.50

    HINT

    数据范围: (2<=M<=9,1<=N<=60), (1<=T<=1000)

    Source

     
    [Submit][Status][Discuss]

    所有的点:

    源点S

    汇点T

    N*M 第i名维修工,倒数第j次维修

    N 第i辆汽车

    所有的边:

    S向N*M个拆分后的维修点,各连一条容量为1,费用为0的边。

    每个汽车点向T连一条容量为1,费用为0的边。

    第i名维修工在倒数第j次维修,修汽车k,造成的代价为 维修时间*j。

    跑最小费用最大流。

      1 #include <bits/stdc++.h>
      2 
      3 #define fread_siz 1024
      4 
      5 inline int get_c(void)
      6 {
      7     static char buf[fread_siz];
      8     static char *head = buf + fread_siz;
      9     static char *tail = buf + fread_siz;
     10 
     11     if (head == tail)
     12         fread(head = buf, 1, fread_siz, stdin);
     13 
     14     return *head++;
     15 }
     16 
     17 inline int get_i(void)
     18 {
     19     register int ret = 0;
     20     register int neg = false;
     21     register int bit = get_c();
     22 
     23     for (; bit < 48; bit = get_c())
     24         if (bit == '-')neg ^= true;
     25 
     26     for (; bit > 47; bit = get_c())
     27         ret = ret * 10 + bit - 48;
     28 
     29     return neg ? -ret : ret;
     30 }
     31 
     32 template <class T>
     33 inline T min(T a, T b)
     34 {
     35     return a < b ? a : b;
     36 }
     37 
     38 const int maxn = 500005;
     39 const int inf = 0x3f3f3f3f;
     40 
     41 int n, m;
     42 int s, t;
     43 int cost;
     44 int edges;
     45 int hd[maxn];
     46 int to[maxn];
     47 int nt[maxn];
     48 int vl[maxn];
     49 int fl[maxn];
     50 
     51 inline void add_edge(int u, int v, int f, int w)
     52 {    
     53     nt[edges] = hd[u]; to[edges] = v; fl[edges] = f; vl[edges] = +w; hd[u] = edges++;
     54     nt[edges] = hd[v]; to[edges] = u; fl[edges] = 0; vl[edges] = -w; hd[v] = edges++;
     55 }
     56 
     57 inline bool bfs(void)
     58 {
     59     static int dis[maxn];
     60     static int que[maxn];
     61     static int inq[maxn];
     62     static int pre[maxn];
     63     static int head, tail;
     64     
     65     memset(inq, 0, sizeof(inq));
     66     memset(dis, inf, sizeof(dis));
     67     
     68     head = 0, tail = 0;
     69     que[tail++] = s;
     70     pre[s] = -1;
     71     dis[s] = 0;
     72     inq[s] = 1;
     73     
     74     while (head != tail)
     75     {
     76         int u = que[head++], v; inq[u] = 0;
     77         
     78         if (head >= maxn)
     79             head = 0;
     80         
     81         for (int i = hd[u]; ~i; i = nt[i])
     82             if (fl[i] && dis[v = to[i]] > dis[u] + vl[i])
     83             {
     84                 dis[v] = dis[u] + vl[i], pre[v] = i^1; 
     85                 if (!inq[v])
     86                     inq[v] = 1, que[tail++] = v;
     87                 if (tail >= maxn)
     88                     tail = maxn;
     89             }
     90     }
     91 
     92     if (dis[t] >= inf)
     93         return false;
     94         
     95     int flow = inf;
     96     
     97     for (int i = pre[t]; ~i; i = pre[to[i]])
     98         if (flow > fl[i ^ 1])
     99             flow = fl[i ^ 1];
    100             
    101     for (int i = pre[t]; ~i; i = pre[to[i]])
    102         fl[i] += flow, fl[i ^ 1] -= flow;
    103         
    104     cost += flow * dis[t];
    105         
    106     return true;
    107 }
    108 
    109 inline void max_flow(void)
    110 {
    111     while (bfs());
    112 }
    113 
    114 signed main(void)
    115 {
    116     n = get_i();
    117     m = get_i();
    118     
    119     s = 0, t = n*m + m + 1;
    120     
    121     memset(hd, -1, sizeof(hd));
    122     
    123     for (int i = 1; i <= n*m; ++i)
    124         add_edge(s, i, 1, 0);
    125     
    126     for (int i = 1; i <= m; ++i)
    127         add_edge(i + n*m, t, 1, 0);
    128         
    129     for (int i = 1; i <= m; ++i)
    130         for (int j = 1; j <= n; ++j)
    131         {
    132             int w = get_i();
    133             for (int k = 1; k <= m; ++k)
    134                 add_edge((j - 1)*m + k, n*m + i, 1, w*k); 
    135         }
    136         
    137     max_flow();
    138     
    139     printf("%.2f
    ", (float)cost / m);
    140 }

    @Author: YouSiki

  • 相关阅读:
    python学习Day21--内置函数、反射
    python学习Day20--属性、类方法和静态方法+类多继承算法补充
    python学习Day19--面向对象的三大特性
    python学习Day18--继承
    python学习Day17--名称空间
    python学习Day16--面向对象
    python学习Day15--递归与二分查找
    python学习Day14--内置函数
    c# 深克隆与浅克隆
    css test-align 和 margin 居中什么区别
  • 原文地址:https://www.cnblogs.com/yousiki/p/6222509.html
Copyright © 2011-2022 走看看