POJ 1275 Cashier Employment 差分约束+二分答案

题意:一家商店在每个小时都需要至少di个人值班,现在有n个人,第j个人可以在fj开始上班,连续工作8个小时,问你要满足商店上班的条件至少需要雇佣多少个人

原题连接:http://poj.org/problem?id=1275

一道比较复杂的查分约束,一开始想着写出每个小时的约束条件,后来发现根本就是牛头不对马嘴。

正解应该是这样:

  设di是第i小时需要的人数,fi是第i小时可以雇佣的最大人数,si是第0-i小时雇佣的人数的和

可以建立这样的约束

  si-si-1 <= fi    人数最多不能超过来应聘的

  si-si-1 >= 0    人数不能为负值

  si-s(i-8)>= di   i>=8 八小时内雇佣的人数大于当前时间点需要的人数

  si-s(i+16)>=di-s24  1<=i<8

但是到了这里约束出现了问题,就是s24是一个未知数,而s24就是我们要求的和,我们要求一个最小的24也就是说

ans <= s24 di - ans >= di - s24,显然要有 si-s(i+16)>=di-ans成立

并且为了保证ans<=s24 s24=s24-s0所以有约束ans<=s24-s0

所以只需要通过枚举ans,然后看差分约束是否可以得到解来判断可行性,枚举的方式用二分即可。

判断可行性使用spfa或者bellmanford就可以

#include <cstdio>
#include <cstring>
#include <cmath>
#include <algorithm>
#include <climits>
#include <string>
#include <iostream>
#include <map>
#include <cstdlib>
#include <list>
#include <set>
#include <queue>
#include <stack>

using namespace std;

typedef long long LL;
const int maxn = 25;
const int maxm = 4e3;
const int INF = INT_MAX / 4;
int first[maxn],nxt[maxm],w[maxm],v[maxm];
int d[maxn],n,cnt[maxn],ecnt,qcnt[maxn],a[maxn];
bool inq[maxn];

void adde(int a,int b,int c) {
    w[ecnt] = c; v[ecnt] = b;
    nxt[ecnt] = first[a];
    first[a] = ecnt;
    ecnt++;
}

bool bellmanford() {
    for(int i = 0;i <= 24;i++) d[i] = INF;
    d[0] = 0;
    for(int i = 0;i <= 24;i++) {
        for(int j = 0;j <= 24;j++) {
            for(int k = first[j];k != -1;k = nxt[k]) {
                if(d[v[k]] > d[j] + w[k]) {
                    d[v[k]] = d[j] + w[k];
                }
            }
        }
    }
    for(int i = 0;i <= 24;i++) {
        for(int j = first[i];j != -1;j = nxt[j]) {
            if(d[v[j]] > d[i] + w[j]) return false;
        }
    }
    return true;
}

bool ok(int sum) {
    int necnt = ecnt,tmpnxt[maxm],tmpfirst[maxn];
    memcpy(tmpnxt,nxt,sizeof(nxt));
    memcpy(tmpfirst,first,sizeof(first));
    for(int i = 0;i < 24;i++) {
        int j = (i + 7) % 24 + 1;
        if(j > i) adde(j,i,-a[j]);
        else adde(j,i,sum-a[j]);
    }
    //adde(0,24,sum);
    adde(24,0,-sum);
    bool ret = bellmanford();
    ecnt = necnt;
    memcpy(nxt,tmpnxt,sizeof(nxt));
    memcpy(first,tmpfirst,sizeof(first));
    return ret;
}

void solve() {
    int l = 0,r = n;
    while(l < r) {
        int mid = (l + r) >> 1;
        //printf("now is %d %d %d\n",l,r,mid);
        if(ok(mid)) r = mid;
        else l = mid + 1;
    }
    if(!ok(l)) puts("No Solution");
    else printf("%d\n",l);
}

int main() {
    //freopen("/tmp/in.txt","r",stdin);
    int T; scanf("%d",&T);
    for(int kase = 1;kase <= T;kase++) {
        ecnt = 0;
        memset(first,-1,sizeof(first));
        memset(nxt,-1,sizeof(nxt));
        memset(cnt,0,sizeof(cnt));
        for(int i = 1;i <= 24;i++) {
            scanf("%d",&a[i]);
        }
        scanf("%d",&n);
        for(int i = 1;i <= n;i++) {
            int str; scanf("%d",&str);
            cnt[str + 1]++;
        }
        for(int i = 1;i <= 24;i++) {
            adde(i - 1,i,cnt[i]);
            adde(i,i - 1,0);
        }
        solve();
    }
    return 0;
}

  

posted @ 2014-07-23 16:22  acm_roll  阅读(209)  评论(0编辑  收藏  举报