题目链接:https://vjudge.net/contest/103424#problem/I

转载于:>>>大牛博客

题目大意:

有 n 个货物,并且知道了每个货物的重量,每次用载重量分别为c1,c2的火车装载,问最少需要运送多少次可以将货物运完。

解题分析:

物品个数最多是10个,可以用0和1分别表示物品是否被选中运送

假设有3个物品,那么可以用001表示当前这一次选择将第3个物品运走

那么,所有的状态可以用0~2^n-1对应的二进制数表示出来

对于上述每一种状态,选择其中可以一次运走的状态进行01背包

其中,所有位全部为1的二进制数表示背包总体积

每个物品的体积是该状态对应二进制数中1的个数

为了把物品全部运走,要选择一些状态把背包的1全部被填满

值得注意的是,因为一件物品不会被运2次,所以所有选取的状态应该是没有交集的

比如1001101和1110010是不行的,第一件物品被运了2次

然后为了使次数最小,可以将次数抽象成01背包中的物品价值,每种状态对应运送一次,价值是1

 

如何判断某种状态是否可以一次运走?

因为有2辆车,所以将要判断该状态是否能分成满足体积分别小于等于车体积的2份,

如何从十进制中取出该状态被选择的物体?

也就是将二进制中的1的位置取出来

利用 :>> :右移运算符 << 左移运算符  & 按位与运算符

假设有一个二进制数: x  =  01101

那么将它右移2位  : x>>=2,他会变成 00011(移出位被丢弃,左边移出的空位或者一律补0,或者补符号位)

我们知道1的二进制数是最后一位为1其他全为0,如果某数的二进制从又往左数第2位是1,移位之后这个1

变成最后一位,和二进制只有最后一位是1的数字1&之后还是1,而如果某数的二进制从又往左数第2位是0

的话,和二进制只有最后一位是1的数字1&之后将是0(因为1除了最后一位其他位全为0,而该数移位后最后一位是0)

综上所述      (x>>i)&1可以判断x从右往左第i位是0还是1【同理(x&(1<<i))也可以判断】

知道怎么找到二进制数中1的位置基本状压的代码也不难懂了

#include<iostream>
#include<algorithm>
#include<string>
#include<cstring>
#include<cstdio>
#include<cmath>
#define mem(a,x) memset(a,x,sizeof(a))
#define inf 1<<30
using namespace std;
const double PI = acos(-1.0);
typedef long long LL;
int state[];
int dp[];
bool vis[];
int n,v1,v2,tot;
int c[];
bool ok(int x)//判断一种状态是否可行(可以一次运走)
{
int sun = ;
mem(vis,);
vis[] = ;
for (int i = ;i < n;i++)
{
if ((x>>i)&)
{
sun += c[i];
for (int j = v1;j >= c[i];j--) //这个真的非常巧妙 开始看半天都不懂,自己模拟一遍才懂
{ //比如说此状态有c1、c2、c3,3个体积,第一次操作把体积c1标记为1,
if (vis[j-c[i]]) //第二次操作把c2和c1+c2两种体积标记为1,第三次把c3和前面的组合标记为1,
vis[j] = ; //最后这些体积能组合成的所有体积就都被标记成了1
}
}
}
if (sun > v1+v2)//装不下
return ;
//总体积小不代表一定装得下,拆分成2份要2份都装得下
for (int i = ;i <= v1;i++)
{
if (vis[i]&&sun-i <= v2)//如果存在(i,sun-i)这样的组合
return ; //满足i可以被v1装下(前面for循环是对于v1的,vis[i]表示体积i可以被v1装下),sun-i可以被v2装下
}
return ;
}
void init()//初始化找到满足条件的状态
{
tot = ;
for (int i = ;i < (<<n);i++)
{
dp[i] = inf;
if (ok(i))
state[tot++] = i;
}
}
int main()
{
int T;
cin>>T;
int oo = ;
while (T--)
{
cin>>n>>v1>>v2;
for (int i = ;i < n;i++)
scanf("%d",&c[i]);
init();
int V = (<<n) - ;//V是n个1的二进制数
dp[] = ;//没有物品当然是0次运走
for (int i = ;i < tot;i++)
{
for (int j = V;j >= ;j--)
{
if (dp[j] == inf)
continue; //原版的背包是dp[j] = min(dp[j],dp[j-c[i]]+w[i])
//但是显然二进制不好表示减,但是可以用|抽象加
//这就相当于背包改版成dp[j+c[i]] = min(dp[j+c[i]],dp[j] + w[i])
if ((j&state[i])==) //当然2种状态不能有交集
{
dp[j|state[i]] = min(dp[j|state[i]] ,dp[j] + );
} }
}
printf("Scenario #%d:\n%d\n",++oo,dp[V]);
if (T) puts("");
}
return ;
}

2018-05-13

POJ 2923 Relocation 装车问题 【状态压缩DP】+【01背包】的更多相关文章

  1. POJ 1691 Painting a Board(状态压缩DP)

    Description The CE digital company has built an Automatic Painting Machine (APM) to paint a flat boa ...

  2. poj 3311 floyd+dfs或状态压缩dp 两种方法

    Hie with the Pie Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 6436   Accepted: 3470 ...

  3. POJ 2686_Traveling by Stagecoach【状态压缩DP】

    题意: 一共有m个城市,城市之间有双向路连接,一个人有n张马车票,一张马车票只能走一条路,走一条路的时间为这条路的长度除以使用的马车票上规定的马车数,问这个人从a出发到b最少使用时间. 分析: 状态压 ...

  4. HDU 6125 Free from square 状态压缩DP + 分组背包

    Free from square Problem Description There is a set including all positive integers that are not mor ...

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

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

  6. poj 2411 Mondriaan's Dream_状态压缩dp

    题意:给我们1*2的骨牌,问我们一个n*m的棋盘有多少种放满的方案. 思路: 状态压缩不懂看,http://blog.csdn.net/neng18/article/details/18425765 ...

  7. poj 1185 炮兵阵地 [经典状态压缩DP]

    题意:略. 思路:由于每个大炮射程为2,所以如果对每一行状态压缩的话,能对它造成影响的就是上面的两行. 这里用dp[row][state1][state2]表示第row行状态为state2,第row- ...

  8. POJ 1038 Bug Integrated Inc(状态压缩DP)

    Description Bugs Integrated, Inc. is a major manufacturer of advanced memory chips. They are launchi ...

  9. poj 2411 Mondriaan's Dream(状态压缩dP)

    题目:http://poj.org/problem?id=2411 Input The input contains several test cases. Each test case is mad ...

随机推荐

  1. 第16月第26天 /bin/bash^M: bad interpreter: 没有那个文件或目录

    1. 运行脚本时出现了这样一个错误,打开之后并没有找到所谓的^M,查了之后才知道原来是文件格式的问题,也就是linux和windows之间的不完全兼容...具体细节不管,如果验证: vim test. ...

  2. 第16月底18天 phpstudy设置

    1.phpstudy设置-端口常规设置 E:\phpStudy\Apache\bin>httpd.exeAH00526: Syntax error on line 14 of E:/phpStu ...

  3. Maven打包编译找不到com.sun.crypto.provider.SunJCE类

    Maven配置 <plugin> <groupId>org.apache.maven.plugins</groupId> <artifactId>mav ...

  4. strstr函数字符串匹配问题

    题目链接:http://acm.sdut.edu.cn/onlinejudge2/index.php/Home/Index/problemdetail/pid/2772.html AC代码: #inc ...

  5. 个人经验~mysql故障处理思路

    一 简介:个人经验总结 二 思路: 从整体上再进行梳理 三 linux角度 1 硬件是否有问题  常见主板 raid卡 和raid磁盘组 2 综合指标 负载 uptime : 1min  5min  ...

  6. [转]OpenBLAS项目与矩阵乘法优化

    课程内容 OpenBLAS项目介绍 矩阵乘法优化算法 一步步调优实现 以下为公开课完整视频,共64分钟: 以下为公开课内容的文字及 PPT 整理. 雷锋网的朋友们大家好,我是张先轶,今天主要介绍一下我 ...

  7. NSURLResponse下载

    // // ViewController.m // 05-NSURLConnestion(下载) // // Created by jerry on 15/10/24. // Copyright (c ...

  8. Window和document的区别

    1.window 窗口对象.就是可视化区域的大小,不包含滚动条内东东. 2.document 对象,包含滚动条以外的区域

  9. Linux命令:pigz多线程压缩工具【转】

    学习Linux系统时都会学习这么几个压缩工具:gzip.bzip2.zip.xz,以及相关的解压工具.关于这几个工具的使用和相互之间的压缩比以及压缩时间对比可以看:Linux中归档压缩工具学习 那么P ...

  10. zabbix通过简单shell命令监控elasticsearch集群状态

    简单命令监控elasticsearch集群状态 原理: 使用curl命令模拟访问任意一个es节点可以反馈的集群状态,集群的状态需要为green curl -sXGET http://serverip: ...