POJ 2923 Relocation(01背包+状态压缩)
题意:有人要搬家,有两辆车可以运送,有若干家具,车有容量限制,而家具也有体积,那么如何运送会使得运送车次最少?规定两车必须一起走,两车一次来回只算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背包+状态压缩)的更多相关文章
- POJ 2923 【01背包+状态压缩/状压DP】
题目链接 Emma and Eric are moving to their new house they bought after returning from their honeymoon. F ...
- POJ 2923 Relocation(01背包变形, 状态压缩DP)
Q: 如何判断几件物品能否被 2 辆车一次拉走? A: DP 问题. 先 dp 求解第一辆车能够装下的最大的重量, 然后计算剩下的重量之和是否小于第二辆车的 capacity, 若小于, 这 OK. ...
- hdu6149 Valley Numer II 分组背包+状态压缩
/** 题目:hdu6149 Valley Numer II 链接:http://acm.hdu.edu.cn/showproblem.php?pid=6149 题意: 众所周知,度度熊非常喜欢图. ...
- hdu6125 Free from square 分组背包+状态压缩
/** 题目:hdu6125 Free from square 链接:http://acm.hdu.edu.cn/showproblem.php?pid=6125 题意: 从不大于n的所有正整数中选出 ...
- poj - 3254 - Corn Fields (状态压缩)
poj - 3254 - Corn Fields (状态压缩)超详细 参考了 @外出散步 的博客,在此基础上增加了说明 题意: 农夫有一块地,被划分为m行n列大小相等的格子,其中一些格子是可以放牧的( ...
- POJ 2923 Relocation (状态压缩,01背包)
题意:有n个(n<=10)物品,两辆车,装载量为c1和c2,每次两辆车可以运一些物品,一起走.但每辆车物品的总重量不能超过该车的容量.问最少要几次运完. 思路:由于n较小,可以用状态压缩来求解. ...
- POJ 2923 Relocation 装车问题 【状态压缩DP】+【01背包】
题目链接:https://vjudge.net/contest/103424#problem/I 转载于:>>>大牛博客 题目大意: 有 n 个货物,并且知道了每个货物的重量,每次用 ...
- [POJ 2923] Relocation (动态规划 状态压缩)
题目链接:http://poj.org/problem?id=2923 题目的大概意思是,有两辆车a和b,a车的最大承重为A,b车的最大承重为B.有n个家具需要从一个地方搬运到另一个地方,两辆车同时开 ...
- POJ-2923 Relocation---01背包+状态压缩
题目链接: https://vjudge.net/problem/POJ-2923 题目大意: 有n个货物,给出每个货物的重量,每次用容量为c1,c2的火车运输,问最少需要运送多少次可以将货物运完 思 ...
随机推荐
- 流媒体中ffmpeg 命令的使用
在linux系统中,使用到的有关流媒体音视频流进行处理的ffmpeg 命令的常用的命令己命令对应的参数如下:记录一下: 1.分离视频音频流 ffmpeg -i input_file -vcodec c ...
- DOM,API,CSS,href,万方db文章,数据库工程师要求
DOM,文档对象模型(Document Object Model),是W3C组织推荐的处理可扩展标志语言的标准编程接口. API,应用程序接口 (API:Application Program Int ...
- day4 函数重载
函数的重载 1.函数重载的定义:在同一个类中,有一个以上的同名函数,只要函数的参数列表或参数类型不一样即可,与返回值无关, 这些统称为方法的重载. 2.函数的重载存在的原因:为了增强方法的阅读性,优化 ...
- 【机器学习】k近邻算法(kNN)
一.写在前面 本系列是对之前机器学习笔记的一个总结,这里只针对最基础的经典机器学习算法,对其本身的要点进行笔记总结,具体到算法的详细过程可以参见其他参考资料和书籍,这里顺便推荐一下Machine Le ...
- ACM-ICPC2018焦作网络赛 Mathematical Curse(dp)
Mathematical Curse 22.25% 1000ms 65536K A prince of the Science Continent was imprisoned in a cast ...
- ASP.NET学习笔记(二)语法
基本的 ASP 语法规则 通常情况下,ASP 文件包含 HTML 标签,类似 HTML 文件.不过,ASP 文件也能够包含服务器端脚本,这些脚本被分隔符 <% 和 %> 包围起来. 在 A ...
- Maven修改默认仓库为阿里云仓库
Maven 仓库默认在国外, 国内使用难免很慢,我们可以更换为阿里云的仓库. 第一步:修改 maven 根目录下的 conf 文件夹中的 setting.xml 文件,在 mirrors 节点上,添加 ...
- CentOS 6.5 安装Clang 3.5.0
来自引用: http://www.cnblogs.com/dudu/p/4294374.html 编译llvm几乎耗费了1个小时-.. 编译CoreCLR需要Clang 3.5,而CentOS上安装的 ...
- 洛谷P1633 二进制
P1633 二进制 题目描述 有三个整数A.B.C,以下用N(2)表示N的二进制(没有前导0). 设A(2).B(2).C(2)的最大长度为L,你需要构造三个正整数X.Y.Z,满足以下条件: (1) ...
- codevs 3162 抄书问题
3162 抄书问题 题目描述 Description 现在要把M本有顺序的书分给K个人复制(抄写),每一个人的抄写速度都一样,一本书不允许给两个(或以上)的人抄写,分给每一个人的书,必须是连续的,比如 ...