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的火车运输,问最少需要运送多少次可以将货物运完 思 ...
随机推荐
- shader之texture
纹理坐标作为属性传递到顶点着色器 texture是OPENGL对象,包含一张或多张相同格式的图片. 它有2中用途: the source of a texture access from a Shad ...
- 文本主题抽取:用gensim训练LDA模型
得知李航老师的<统计学习方法>出了第二版,我第一时间就买了.看了这本书的目录,非常高兴,好家伙,居然把主题模型都写了,还有pagerank.一路看到了马尔科夫蒙特卡罗方法和LDA主题模型这 ...
- windows和Dos常见命令总结
linux最常见命令 (1) pwd命令pwd (即print working directory,打印工作路径) 命令的功能是显示当前的工作路径.如现在是在“/home/CAI”目录下,则可以用此命 ...
- 同余方程 (codevs1200)
题目描述××× 求关于 x 的同余方程 ax ≡ 1 (mod b)的最小正整数解. 输入输出格式××× 输入格式: 输入只有一行,包含两个正整数 a, b,用一个空格隔开. 输出格式: 输出只有一行 ...
- 阿里云(centos)下svn 服务器搭建
安装说明 系统环境:阿里云centos安装方式:yum install subversion 检查已安装版本 #检查是否安装了低版本的SVN[root@localhost /]# rpm -qa su ...
- PJzhang:子域名发掘工具Sublist3r
猫宁!!! 参考链接:https://www.freebuf.com/sectool/90584.html 作者上一次更新是2018年10月16日了,sublist3r中融合有另外一个子域名爆破工具S ...
- jQuery EasyUI/TopJUI创建日期时间输入框
jQuery EasyUI/TopJUI创建日期时间输入框 日期时间输入框组件 HTML 和日期输入框类似,日期时间输入框允许用户选择日期和指定的时间并按照指定的输出格式显示.相比日期输入框,它在下拉 ...
- 怎么解决java.lang.NoClassDefFoundError错误
http://blog.csdn.net/jamesjxin/article/details/46606307
- JSONPath中的表达式
在JsonPath中使用表达式是一个非常好的功能,可以使用简洁和复杂的JsonPath.JsonPath中的表达式基本上是评估为布尔值的代码片段.基于结果,仅选择满足标准的节点.让我们看一下它的更多内 ...
- 2.排序检索数据 ---SQL
order by 一.排序数据 SELECT prod_name FROM Products ORDER BY prod_name; ORDER BY子句的位置 在指定一条ORDER BY子句时,应该 ...