POJ-2923 Relocation---01背包+状态压缩

时间:2023-02-13 00:03:42

题目链接:

https://vjudge.net/problem/POJ-2923

题目大意:

有n个货物,给出每个货物的重量,每次用容量为c1,c2的火车运输,问最少需要运送多少次可以将货物运完

思路:

第一次做状态压缩(状态压缩基础知识传送门

本题的解题思路是先枚举选择若干个时的状态,总状态量为1<<n,判断这些状态集合里的那些物品能否一次就运走,如果能运走,那就把这个状态看成一个物品。预处理完能从枚举中找到tot个物品,再用这tol个物品中没有交集(也就是两个状态不能同时含有一个物品)的物品进行01背包,每个物品的体积是state[i](state[i]表示一次可以运完状态i的物品,i的二进制表示i这个状态的物品),价值是1,求包含n个物品的最少价值也就是dp[(1<<n)-1](dp[i]表示状态i需要运的最少次数)。

状态转移方程:dp[j|k] = min(dp[j|k],dp[k]+1) (k为state[i],1<=j<=(1<<n)-1])。

 #include<iostream>
#include<vector>
#include<queue>
#include<algorithm>
#include<cstring>
#include<cstdio>
#include<set>
#include<cmath>
using namespace std;
typedef pair<int, int> Pair;
typedef long long ll;
const int INF = 0x3f3f3f3f;
const int maxn = +;
int T, n, m1, m2;
int a[], cnt[maxn], dp[maxn], tot, cases;
bool vis[maxn];
bool judge(int x)
{
int sum = ;
memset(vis, , sizeof(vis));//vis[i]=1表示m1的车子中可以凑出体积为i的物品
vis[] = ;
for(int i = ; i < n; i++)
{
if(x & ( << i))//第i件物品存在
{
sum += a[i];
for(int j = m1; j >= a[i]; j--)
if(vis[j - a[i]])vis[j] = ;//此处必须是逆序,因为更新vis[j]的时候要用到vis[j-a[i]],和01背包是一样的
}
}
for(int i = ; i <= m1; i++)
{
if(vis[i] && sum - i <= m2)//确保全部物品可以一次性放在两个车子里面
return true;
}
return false;
}
void init()
{
memset(dp, INF, sizeof(dp));
dp[] = ;
for(int i = ; i < ( << n); i++)
{
if(judge(i))
{
cnt[tot++] = i;
}
}
}
int main()
{
cin >> T;
while(T--)
{
cin >> n >> m1 >> m2;
tot = ;
for(int i = ; i < n; i++)cin >> a[i];
init();/*
for(int i = 0; i < tot; i++)
cout<<cnt[i]<<endl;*/
for(int i = ; i < tot; i++)//枚举物品
{
for(int j = ( << n) - ; j >= ; j--)//逆序枚举状态也是因为dp[j]的更新需要先用到dp[j-***]
{
if(dp[j] == INF)continue;
if((j & cnt[i]) == )//两者无交集
dp[j | cnt[i]] = min(dp[j | cnt[i]], dp[j] + );
//dp[j | cnt[i]]表示j这个状态加上第i件物品的值,可以从dp[j]+1推过去
}
}
printf("Scenario #%d:\n", ++cases);
printf("%d\n\n", dp[(<<n) - ]); }
}

POJ-2923 Relocation---01背包+状态压缩的更多相关文章

  1. POJ 2923 【01背包&plus;状态压缩&sol;状压DP】

    题目链接 Emma and Eric are moving to their new house they bought after returning from their honeymoon. F ...

  2. POJ 2923 Relocation&lpar;01背包变形&comma; 状态压缩DP&rpar;

    Q: 如何判断几件物品能否被 2 辆车一次拉走? A: DP 问题. 先 dp 求解第一辆车能够装下的最大的重量, 然后计算剩下的重量之和是否小于第二辆车的 capacity, 若小于, 这 OK. ...

  3. POJ 2923 Relocation(01背包&plus;状态压缩)

    题意:有人要搬家,有两辆车可以运送,有若干家具,车有容量限制,而家具也有体积,那么如何运送会使得运送车次最少?规定两车必须一起走,两车一次来回只算1躺. 思路:家具怎么挑的问题,每趟车有两种可能:1带 ...

  4. hdu6149 Valley Numer II 分组背包&plus;状态压缩

    /** 题目:hdu6149 Valley Numer II 链接:http://acm.hdu.edu.cn/showproblem.php?pid=6149 题意: 众所周知,度度熊非常喜欢图. ...

  5. hdu6125 Free from square 分组背包&plus;状态压缩

    /** 题目:hdu6125 Free from square 链接:http://acm.hdu.edu.cn/showproblem.php?pid=6125 题意: 从不大于n的所有正整数中选出 ...

  6. poj - 3254 - Corn Fields (状态压缩)

    poj - 3254 - Corn Fields (状态压缩)超详细 参考了 @外出散步 的博客,在此基础上增加了说明 题意: 农夫有一块地,被划分为m行n列大小相等的格子,其中一些格子是可以放牧的( ...

  7. POJ 2923 Relocation (状态压缩,01背包)

    题意:有n个(n<=10)物品,两辆车,装载量为c1和c2,每次两辆车可以运一些物品,一起走.但每辆车物品的总重量不能超过该车的容量.问最少要几次运完. 思路:由于n较小,可以用状态压缩来求解. ...

  8. POJ 2923 Relocation 装车问题 【状态压缩DP】&plus;【01背包】

    题目链接:https://vjudge.net/contest/103424#problem/I 转载于:>>>大牛博客 题目大意: 有 n 个货物,并且知道了每个货物的重量,每次用 ...

  9. &lbrack;POJ 2923&rsqb; Relocation &lpar;动态规划 状态压缩&rpar;

    题目链接:http://poj.org/problem?id=2923 题目的大概意思是,有两辆车a和b,a车的最大承重为A,b车的最大承重为B.有n个家具需要从一个地方搬运到另一个地方,两辆车同时开 ...

随机推荐

  1. yum简单安装apache

    yum install httpd -y chkconfig    httpd  on service httpd start  启动软件

  2. Android UI开发第四十篇——ScrollTricks介绍

    ScrollTricks是一个开源控件,实现了两个简单功能: 1.Quick Return:向上滑动时,View也向上滑动并且消失,当向下滑动时,View马上出现.例如Google Now的搜索功能. ...

  3. Jedis的JedisSentinelPool源代码分析

    概述 Jedis是Redis官方推荐的Java客户端,更多Redis的客户端可以参考Redis官网客户端列表.Redis-Sentinel作为官方推荐的HA解决方案,Jedis也在客户端角度实现了对S ...

  4. Android 风格化的 Toggle Buttons

    Android到默认UI比iOS到默认UI在美观程度上还是有一定到差距的,我们希望能够美化UI,并且替换掉系统默认的UI风格,使得程序在使用这些UI的时候都默认使用我们自定义到UI.本文以Toggle ...

  5. 洛谷 P3392 涂国旗

    P3392 涂国旗 题目描述 某国法律规定,只要一个由N*M个小方块组成的旗帜符合如下规则,就是合法的国旗.(毛熊:阿嚏——) 从最上*干行(>=1)的格子全部是白色的. 接下来若干行(&gt ...

  6. 微信支付生成带logo的二维码

    利用到一个qrcode类 比较简洁 原作者没有加入二维码嵌入logo的功能 在这里我进行了小小的修改 可以实现生成微信支付二维码时打上logo 生成png格式的利用到该类中的png方法(我已经改好了) ...

  7. 【English】三、以o结尾单词变复数

    一.以O结尾的词,许多加es构成复数,特别是一些常用词如: potatoes      土豆 tomatoes     西红柿 echoes 回声 tornadoes    龙卷风 torpedoes ...

  8. python数据类型&lpar;二&rpar;

    跟着慕课网练习的,一些简单的知识点如下

  9. Oracle中查询表的大小、表的占用情况和表空间的大小

    有两种含义的表大小.一种是分配给一个表的物理空间数量,而不管空间是否被使用.可以这样查询获得字节数: select segment_name, bytes from user_segments whe ...

  10. URAL-1019 Line Painting----暴力或线段树

    题目链接: https://cn.vjudge.net/problem/URAL-1019 题目大意: 一个0~1e9的区间,初始都是白的,现进行N次操作,每次将一段区间图上一中颜色.最后问说连续最长 ...