poj 2923 Relocation (枚举+背包 | 状态压缩+01背包)

来源:互联网 发布:系统动力学软件 价格 编辑:程序博客网 时间:2024/05/01 07:59

链接:

http://poj.org/problem?id=2923


题目大意:

有N件家具,每件的重量为(1 ≤ wi ≤ 100), 用两辆车把他们来运送到目的地。这两辆车的限载重量分别为C1, C2(1 ≤ Ci ≤ 100) , 问最少几趟可以把所有家具运送到目的地。

这两辆车每趟都必须一起走,即使某辆车没载东西。


思路:

(一)

先上自己的方法:

枚举第一辆车可能载的家具的所有组合情况,那么用二进制来表示状态则共有 1<<n ,如果二进制的第i位上是1,那么就表示第i件家具是由第一辆车运的。

这样就相当于把所有家具分成了两组,一组给第一辆车运,另一组给第二辆车运。

然后分别对这两组数据计算,分别最少几次可以运完,这样,问题就转换为了:给n个家具,和一辆限载重量c的车,这辆车最少几次可以运完。

由贪心的思想可以知道,每次运送应该尽可能的装满这辆车,那么用01背包,背包容量为限载重量c,每个家具的重量即是价值也是耗费, f[c]就是这次运送的。选择完一次后,把那些选择过的家具删除掉,继续再做01背包选择,直到所有家具全部选择完。至于记录选择了哪些家具,方法和记录路径一样。

然后,这两辆车分别运送次数取较大的,就是这个枚举到的这个方案的最少次数。


(二)

这个方法是学习的,网上的基本都是这个方法,不再细讲。

大概思路是,用二进制表示选择状态,先预处理所有选择的组合,看这个组选择的家具能否被两辆车子一次运过去,如果可以的话就“打包”起来看作是一个物体,然后再对所有物体进行01背包。

 


代码:

方法一:

#include<iostream>#include<queue>#include<cstdio>#include<cstring>#include<cmath>#include<map>#include<vector>#include<string>#define MP make_pair#define SQ(x) ((x)*(x))const double PI = acos(-1.0);const int INF = 0x3f3f3f3f;using namespace std;typedef long long int64;const int MAXN = 12;int n, m;int c1, c2;int f[110];int w[MAXN];int ans;bool path[MAXN][110];vector<int>w1, w2;bool check(int x){    for(int i=0; i<n; ++i){        if((x>>i)&1){            if(w[i]>c1) return false;        }else{            if(w[i]>c2) return false;        }    }     return true;}void init(int x){    w1.clear(); w2.clear();    for(int i=0; i<n; ++i){        if((x>>i)&1){            w1.push_back(w[i]);        }else            w2.push_back(w[i]);    }}void update(vector<int>&w, int i, int j){    if(i<0) return ;    int tmp = w[i];    if(path[i][j]){        w.erase(w.begin()+i);        update(w, i-1, j-tmp);    }else        update(w, i-1, j);}int counter(vector<int>& w, int c){    if(w.size()==0)  return 0;       int cnt = 0;    do{        cnt++;        if(cnt >= ans) return cnt;        memset(f, 0, sizeof(f));        memset(path, 0, sizeof(path));                for(int i=0; i<w.size(); ++i){            for(int v=c; v>=w[i]; --v){                if(f[v-w[i]]+w[i] >= f[v]){                    f[v] = f[v-w[i]]+w[i];                    path[i][v] = 1;                }            }        }        update(w, w.size()-1, c);    }while(w.size());    return cnt;}int main(){    int nCase, cas=1, x;    scanf("%d", &nCase);    while(nCase--){        printf("Scenario #%d:\n", cas++);        scanf("%d%d%d", &n,&c1,&c2);        for(int i=0; i<n; ++i)            scanf("%d", &w[i]);        ans = INF;        int sta = 0;        while(sta < (1<<n)){            if(!check(sta)){                sta++;                continue;             }            init(sta);            int cnt1 = counter(w1, c1);            int cnt2 = counter(w2, c2);            ans = min(ans, max(cnt1, cnt2));            sta++;        }        printf("%d\n\n", ans);    }    return 0;} 


方法二:

#include<iostream>#include<queue>#include<cstdio>#include<cstring>#include<cmath>#include<map>#include<vector>#include<string>#define MP make_pair#define SQ(x) ((x)*(x))const double PI = acos(-1.0);const int INF = 0x3f3f3f3f;using namespace std;typedef long long int64;const int MAXN = 15;int n, m;int c1, c2;int w[MAXN], d[2000];int sta[2000], idx;bool check(int x){    int sum = 0 ;    for(int i=0; i<=c1; ++i) d[i] = 0;    d[0] = 1;    for(int i=0; i<n; ++i) if((x>>i)&1){         sum += w[i];        for(int v=c1; v>=w[i]; --v)            d[v] = max(d[v], d[v-w[i]]+w[i]);    }    if(sum > c1+c2) return false;    return sum-d[c1] <= c2;}int main(){    int nCase, cas=1, x;    scanf("%d", &nCase);    while(nCase--){        printf("Scenario #%d:\n", cas++);        scanf("%d%d%d", &n,&c1,&c2);        for(int i=0; i<n; ++i)            scanf("%d", &w[i]);        idx = 0;        int i = 1;        while(i < (1<<n)){            if(check(i)){                sta[idx++] = i;            }            ++i;        }        for(int i=0; i<(1<<n); ++i) d[i]=INF;        d[0] = 0;        for(int i=0; i<idx; ++i){            for(int j=((1<<n)-1); j>=0; --j){                if(d[j] == INF) continue;                if((j&sta[i]) == 0){                    d[j|sta[i]] = min(d[j|sta[i]], d[j]+1);                }            }        }        printf("%d\n\n", d[(1<<n)-1]);    }    return 0;} 









原创粉丝点击