题意:有人要搬家,有两辆车可以运送,有若干家具,车有容量限制,而家具也有体积,那么如何运送会使得运送车次最少?规定两车必须一起走,两车一次来回只算1躺。

思路:家具怎么挑的问题,每趟车有两种可能:1带走此家具;2此趟暂时不带走。那就是01背包了。但是限制是两只车的容量。求的是趟数。

1)数据量较小,将这10件以内的所有物品的可能组合记录一下,有2^10种,其中是包含了一些运不走的组合,滤掉,只留下可能的组合。对每种可能的新组合进行01背包,即考虑对于新组合i这种运送方案该不该取,如果取了能使车次更少,那就取。那就得穷举除了i所组合的物品外的所有可能的组合,所要决策的是 “该组合与i组合”后有没有可能减少车次,没有的话不更新了,按老方案。

 #include <iostream>
#include <cstring>
#include <cstdio>
#define INF 0x1fffffff
using namespace std;
const int N=;
int n,car1,car2;
int w[N];
int tmp_dp[<<N];
int pre[<<N];
int dp[<<N]; int can_take(const int j)
{
int sum=;
memset(tmp_dp,,sizeof(tmp_dp));
for(int i=; i<n; i++)
{
if( (<<i)&j )
{
sum+=w[i];
if(car1+car2<sum)
return ;
for(int k=car1; k>=w[i]; k-- )//对其中一只车进行01背包
tmp_dp[k]=max(tmp_dp[k], tmp_dp[k-w[i]]+w[i]);
}
}
if(sum-tmp_dp[car1]>car2) return ;
return ;
} int cal()
{
memset(pre,,sizeof(pre));
int ful=(<<n)-, len=;
for(int i=ful; i>; i-- ) //预处理
if( can_take(i) )
pre[len++]=i;
int sta=;
dp[]=;
for(int i = ; i<=ful; i++) dp[i] = INF; //初始化 for(int i=; i<len; i++) //每个组合品
{
for(int j=ful-pre[i]; j>=; j--) //扣除i这几种物品,穷举其他所有的组合品(包括空组合),看是否与组合品i组合会使用更少的次数。如果原来已经有方案运走包括组合品i与其他一些组合的车次更少,那么不考虑运送组合品i(因为i组合得不合理)。
{
if( !(j&pre[i])) //j和组合品i无交集,在原来的方案j上考虑第i个放不放,若放就将车次+1。如果放,则要更新的应该是j|pre[i]这个放了i组合品的状态。
{
dp[j|pre[i]]=min(dp[j]+, dp[j|pre[i]]); //(放, 不放)
}
}
}
return dp[ful];
} int main() {
freopen("input.txt", "r", stdin);
int t, e=;
cin>>t;
while(t--)
{
scanf("%d %d %d", &n, &car1, &car2);
for(int i=; i<n; i++) scanf("%d", &w[i]);
printf("Scenario #%d:\n%d\n\n", ++e, cal());
}
return ;
}

AC代码

2) WA思路:对所有可能运走的组合计算其最大运送量并记录其状态,每步从中找出不与之前相交的最大运送量,看几趟之后能全送走。贪心的思想,每步择运送量最大的,只要维持没有交集就行了,最后肯定全都送走。过了样例,过不了poj的数据。难道还有一种运法:每一趟不是最佳,但是和下一趟组合起来就是最佳?比如有两趟的方案100+60,但也有方案80+70,这样就变成每趟非最佳,但是总方案却是最佳。MYBE!

 #include <iostream>
#include <cstring>
#include <cstdio>
using namespace std;
const int N=;
int n,car1,car2;
int w[N];
int tmp_dp[<<N];
int pre[<<N]; int can_take(const int j)
{
int sum=;
memset(tmp_dp,,sizeof(tmp_dp));
for(int i=; i<n; i++)
{
if( (<<i)&j )
{
sum+=w[i];
if(car1+car2<sum)
return ;
for(int k=car1; k>=w[i]; k-- )
tmp_dp[k]=max(tmp_dp[k], tmp_dp[k-w[i]]+w[i]);
}
}
if(sum-tmp_dp[car1]>car2) return ;
return sum;
} int cal()
{
memset(pre,,sizeof(pre));
int ful=(<<n)-, len=; for(int i=ful; i>; i-- ) //预处理:pre[i]表示i这种组合的家具价值
pre[i]=can_take(i); int sta=;
for(int i=; i<n; i++) //最多n趟
{
int v=,s=;
for(int j=ful; j>=; j--) //扫描除了sta的所有组合,挑出运送量最大的。
{
if(!(j&sta)) //无交集
{
if(pre[j]>v)
{
v=pre[j];
s=j;
}
}
}
sta|=s;
if((sta&ful)==ful) //如果已经运完
return i+;
}
return n;
} int main() {
//freopen("input.txt", "r", stdin);
int t, e=;
cin>>t;
while(t--)
{
scanf("%d %d %d", &n, &car1, &car2);
for(int i=; i<n; i++) scanf("%d", &w[i]); printf("Scenario #%d:\n%d\n\n", ++e, cal());
}
return ;
}

WA代码

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

  1. POJ 2923 【01背包+状态压缩/状压DP】

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

  2. POJ 2923 Relocation(01背包变形, 状态压缩DP)

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

  3. hdu6149 Valley Numer II 分组背包+状态压缩

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

  4. hdu6125 Free from square 分组背包+状态压缩

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

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

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

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

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

  7. POJ 2923 Relocation 装车问题 【状态压缩DP】+【01背包】

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

  8. [POJ 2923] Relocation (动态规划 状态压缩)

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

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

    题目链接: https://vjudge.net/problem/POJ-2923 题目大意: 有n个货物,给出每个货物的重量,每次用容量为c1,c2的火车运输,问最少需要运送多少次可以将货物运完 思 ...

随机推荐

  1. ubuntu下caffe配置

    http://blog.csdn.net/yhaolpz/article/details/71375762

  2. asn编译常见报错

    TypeError: unsupported operand type(s) for -: 'str' and 'int' 可能是该用列表的地方没用列表. 1. ’-‘不支持,需改为'_' asn1t ...

  3. 7.22实习培训日志-JSP Servlet

    周末总结 今天下午在学习servlet,想自己做一个简单的例子,于是用idea新建一个maven项目,为了后文叙述方便,我们取名为项目1,点击create from archetype,我先选择org ...

  4. CSS 定位 (Positioning)概述

    div.h1 或 p 元素常常被称为块级元素. 这意味着这些元素显示为一块内容,即“块框”. 与之相反,span 和 strong 等元素称为“行内元素”,这是因为它们的内容显示在行中,即“行内框”. ...

  5. memcached 扩展安装(windows)

    在脚本之家里下载扩展压缩包 https://www.jb51.net/softs/392873.html 安装扩展前得先安装memcached并启动 下载完扩展压缩包解压并找到对应自己php版本 复制 ...

  6. 支持通配符查询的k-gram索引

    k-gram索引的通配符查询处理技术称为k-gram索引. 一个k-gram代表由k个字符组成的序列.对于词项castle来说,cas.ast.stl都是3-gram.我们用特殊的字符$来标识词项的开 ...

  7. 进击JavaScript核心 --- (1)基本数据类型

    ES5之前提供了 5种基本数据类型 和 1种引用数据类型 基本数据类型:Undefined, Null, String, Number, Boolean 引用数据类型:Object ES6开始引入了一 ...

  8. Java-GC-标记清除算法

    ## 前置知识 静态变量在类被加载的时候分配内存.当我们启动一个App的时候,系统会创建一个进程,此进程会加载一个JVM的实例,然后代码就运行在JVM之上.也就是说类在被加载的时候,静态变量 --- ...

  9. Centos7.x 安装 pptp

    VPN 1:检查是否支持PPTP #返回OK && echo ok ok 2:安装ppp yum install -y ppp 3:导入EPEL源 rpm -ivh http://dl ...

  10. js 阻止元素发生默认的行为ie兼容

    //亲测ie9可以兼容 function Go(event) { var e = event; if (e && e.preventDefault) { e.preventDefaul ...