洛谷P2050 [NOI2012]美食节

如果没有做过修车(本题的弱化版),可以先去做一下。

大致思路就是把每个厨师拆成\(p\)个点,分别代表倒数第几个做哪道菜,然后从每种菜向厨师连边,倒数第几个的边权就是几倍的等待时间,最后跑一波费用流即可。

但本题的数据范围有些残酷,所以我们考虑动态加边。最多只会跑\(p\)\(spfa\),并且每次走最短路增广,所以很多边都是没有用的。首先把代表倒数第一个做的菜的边加上。然后,比如某一次跑完\(spfa\)后,是沿第\(j\)个厨师的倒数第\(k\)个增广的,那就把他的代表倒数\(k+1\)的边补全。

代码如下:

#include <bits/stdc++.h>

using namespace std;

#define N 40
#define M 100
#define P 800
#define SZ (N+P*M)
#define pb push_back
#define INF 0x3f3f3f3f

struct Edge
{
    int from, to, cap, cost;
};

int n, m, tot, S, T, ans, ans0, t[N+5], d[SZ+5], vis[SZ+5], a[SZ+5], pre[SZ+5], tim[M+5][N+5], lim[M+5];
vector<int> G[SZ+5];
vector<Edge> edges;

int ID(int j, int k)
{
    return n+(j-1)*tot+k;
}

int rev(int id)
{
    return (id-n-1)/tot+1;
}

void addEdge(int u, int v, int cap, int cost)
{
    edges.pb(Edge{u, v, cap, cost}), edges.pb(Edge{v, u, 0, -cost});
    G[u].pb(edges.size()-2), G[v].pb(edges.size()-1);
}

int spfa()
{
    memset(d, 0x3f, sizeof d);
    d[S] = 0, vis[S] = 1, a[S] = INF, pre[S] = 0;
    queue<int> q; q.push(S);
    while(!q.empty())
    {
        int u = q.front(); q.pop();
        vis[u] = 0;
        for(int i = 0, v, w; i < G[u].size(); ++i)
        {
            Edge &e = edges[G[u][i]]; v = e.to, w = e.cost;
            if(e.cap > 0 && d[v] > d[u]+w)
            {
                d[v] = d[u]+w;
                a[v] = min(a[u], e.cap);
                pre[v] = G[u][i];
                if(!vis[v]) q.push(v), vis[v] = 1;
            }
        }
    }
    if(d[T] == INF) return 0;
    int u = T;
    ans0 += a[T], ans += d[T]*a[T];
    while(u != S)
    {
        edges[pre[u]].cap -= a[T], edges[pre[u]^1].cap += a[T];
        u = edges[pre[u]].from;
    }
    return 1;
}

int main()
{
    #ifndef ONLINE_JUDGE
        freopen("testdata.in", "r", stdin);
        freopen("testdata.out", "w", stdout);
    #endif
    scanf("%d%d", &n, &m);
    S = 0;
    for(int i = 1; i <= n; ++i) scanf("%d", &t[i]), tot += t[i], addEdge(S, i, t[i], 0);
    T = n+tot*m+1;
    for(int i = 1; i <= n; ++i)
    {
        for(int j = 1, t; j <= m; ++j) scanf("%d", &tim[j][i]), addEdge(i, ID(j, 1), 1, tim[j][i]), lim[j] = 1;
    }
    for(int i = n+1; i <= n+tot*m; ++i) addEdge(i, T, 1, 0);
    while(spfa())
    {
        int j = rev(edges[pre[T]].from);
        lim[j]++;
        for(int i = 1; i <= n; ++i) addEdge(i, ID(j, lim[j]), 1, lim[j]*tim[j][i]);
    }
    printf("%d\n", ans);
    return 0;
}
posted @ 2018-12-14 08:53  dummyummy  阅读(182)  评论(0编辑  收藏  举报