POJ 2923 Relocation (状态压缩,01背包)
题意:有n个(n<=10)物品,两辆车,装载量为c1和c2,每次两辆车可以运一些物品,一起走。但每辆车物品的总重量不能超过该车的容量。问最少要几次运完。
思路:由于n较小,可以用状态压缩来求解。
家具从左到右依次对应二进制形式的低位到高位,该位上为1表示该家具还没运走,0表示已经运走。
建立两个数组,s1[],s2[],分别存储一次性能被货车1和货车2运走的状态。
之后再把s1,s2数组中的状态合并一下,存入到s数组中去,即表示能一次性被两辆货车同时运走的状态。
合并条件:若(s1&s2)=0,即没有重合的,则两者可以合并
dp[i]表示剩余家具状态为i时最少要搬几次
状态转移方程:
dp[j]=min(dp[j-s[i])+1; //当然(s[i]|j)=0,即s[i]为j的子序列
有两个要注意的地方!!!
1.先用vis数组,若vis[s1|s2]=1表示该状态(s1与s2合并的状态)可以一次性被两辆车运走,
之后再把相应的状态存入s1s2数组。
因为如果直接存储,可能会有几次(s1|s2)数值相同的情况,这样可能在s1s2中存储多个相同的值,这样会对答案有影响。
2.一开始忽略一点,就是有可能运家具只是用其中一辆货车,另一辆装不下任何货物。。。
所以状态要从0开始枚举,而不是从1开始。。。
附上两个代码:
01背包,即直接dp方程求解:
#include <iostream>
#include <stdio.h>
#include <cstring>
#include <algorithm> using namespace std;
const int INF=0x3f3f3f3f;
const int maxn=(<<)+;
int n,c1,c2;
int w[];
//s1存储能一次性被货车1运走的状态,s2存储能一次性被货车2运走的状态,s存储能一次性被两辆货车同时运走的状态
int s1[maxn],s2[maxn],s[maxn];
int vis[maxn]; //标记哪些状态可以一次性被两辆货车运走
int idx1,idx2,idx;
int dp[maxn]; void init(){
idx1=idx2=idx=-;
int sum;
//忽略一点,就是有可能之后运家具只是用其中一辆货车,另一辆装不下任何货物。。。
//所以状态要从0开始。。。
for(int i=;i<(<<n);i++){
sum=;
for(int j=;j<n;j++){
if(i&(<<j)){
sum+=w[j];
}
}
if(sum<=c1)
s1[++idx1]=i;
if(sum<=c2)
s2[++idx2]=i;
}
/*
先用vis数组存储哪些能一次性运走,之后在存入s数组.
因为如果直接存入s数组的话,可能有好几组s1[i]|s2[j]相同,这样同一个数会存好几次,可能会有影响
*/
memset(vis,,sizeof(vis));
for(int i=;i<=idx1;i++){
for(int j=;j<=idx2;j++){
if((s1[i]&s2[j])==){
vis[s1[i]|s2[j]]=;
}
}
}
for(int i=;i<(<<n);i++)
dp[i]=INF;
for(int i=;i<(<<n);i++){
if(vis[i]){
s[++idx]=i;
dp[i]=;
}
}
}
int main()
{
int t;
scanf("%d",&t);
for(int q=;q<=t;q++){
scanf("%d%d%d",&n,&c1,&c2);
for(int i=;i<n;i++){
scanf("%d",&w[i]);
}
init();
for(int i=;i<=idx;i++){
for(int j=(<<n)-;j>=s[i];j--){
if((s[i]|j)==j){
dp[j]=min(dp[j],dp[j-s[i]]+);
}
}
}
printf("Scenario #%d:\n",q);
printf("%d\n\n",dp[(<<n)-]);
}
return ;
}
dfs记忆化搜索:
#include <iostream>
#include <string.h>
#include <algorithm>
#include <stdio.h> using namespace std; int s1[];//存储可以被c1货车一次性搬走的家具的不同状态,二进制下1代表还没搬,0代表已搬走了
int s2[];//存储可以被c2货车一次性搬走的家具的不同状态,二进制下1代表还没搬,0代表已搬走了
int s1s2[];//存储可以一次性被两辆车运载的情况
int vis[];//vis[i]:若i能一次性被两辆车运走,则vis[i]=1;
int index1,index2,index3;
int dp[(<<)+];//dp[i]表示剩余家具状态为i时最少要搬几次
int t,n,c1,c2,load;
int w[];
int ans; void init(){
for(int i=;i<(<<n);i++){
load=; //忘记每次开始的时候load=0了;
for(int p=;p<n;p++){
if(i&(<<p)){
load+=w[p];
}
}
if(load<=c1){
s1[index1]=i;
index1++;
}
if(load<=c2){
s2[index2]=i;
index2++;
}
}
} int dfs(int s){
int ss1,ss2;
int ss;
if(dp[s]!=-)
return dp[s];
for(int i=;i<index3;i++){
if(s1s2[i]==s){
dp[s]=;
return dp[s];
}
}
dp[s]=; for(int i=;i<index3;i++){
ss=s1s2[i];
if((ss|s)==s){
dp[s]=min(dp[s],dfs(s-ss)+);
}
}
return dp[s];
}
int main()
{
scanf("%d",&t);
for(int i=;i<=t;i++){
memset(dp,-,sizeof(dp));
memset(vis,,sizeof(vis));
index1=index2=index3=;
scanf("%d%d%d",&n,&c1,&c2);
for(int j=;j<n;j++)
scanf("%d",&w[j]);
init();
int s11,s22;
//先将可以一次性被两辆车运走的情况给存起来
for(int m=;m<index1;m++){
for(int j=;j<index2;j++){
s11=s1[m];
s22=s2[j];
if((s11&s22)==){
vis[s11|s22]=;
}
}
}
for(int m=;m<(<<n);m++){
if(vis[m]==){
s1s2[index3]=m;
index3++;
}
}
ans=dfs((<<n)-);
printf("Scenario #%d:\n",i);
printf("%d\n\n",ans);
}
return ;
}
POJ 2923 Relocation (状态压缩,01背包)的更多相关文章
- Relocation - POJ 2923(状态压缩+01背包)
题目大意:有个人需要搬家,有N件物品,给个物品的重量是 w[i] 然后又两个车,每个车的载重量分别是C1和C2,求最少需要运输多少次才能把这些物品全部运输完毕. 分析:刚开始就发现物品数不多,想着直接 ...
- poj 2923 状压dp+01背包
好牛b的思路 题意:一系列物品,用二辆车运送,求运送完所需的最小次数,两辆车必须一起走 解法为状态压缩DP+背包,本题的解题思路是先枚举选择若干个时的状态,总状态量为1<<n,判断这些状态 ...
- BZOJ 4197: [Noi2015]寿司晚宴 状态压缩 + 01背包
4197: [Noi2015]寿司晚宴 Time Limit: 10 Sec Memory Limit: 512 MB Description 为了庆祝 NOI 的成功开幕,主办方为大家准备了一场寿 ...
- POJ 2923 Relocation 装车问题 【状态压缩DP】+【01背包】
题目链接:https://vjudge.net/contest/103424#problem/I 转载于:>>>大牛博客 题目大意: 有 n 个货物,并且知道了每个货物的重量,每次用 ...
- POJ 2184 Cow Exhibition【01背包+负数(经典)】
POJ-2184 [题意]: 有n头牛,每头牛有自己的聪明值和幽默值,选出几头牛使得选出牛的聪明值总和大于0.幽默值总和大于0,求聪明值和幽默值总和相加最大为多少. [分析]:变种的01背包,可以把幽 ...
- poj 3311(状态压缩DP)
poj 3311(状态压缩DP) 题意:一个人送披萨从原点出发,每次不超过10个地方,每个地方可以重复走,给出这些地方之间的时间,求送完披萨回到原点的最小时间. 解析:类似TSP问题,但是每个点可以 ...
- poj 1185(状态压缩DP)
poj 1185(状态压缩DP) 题意:在一个N*M的矩阵中,‘H'表示不能放大炮,’P'表示可以放大炮,大炮能攻击到沿横向左右各两格,沿纵向上下各两格,现在要放尽可能多的大炮使得,大炮之间不能相互 ...
- poj 3254(状态压缩DP)
poj 3254(状态压缩DP) 题意:一个矩阵里有很多格子,每个格子有两种状态,可以放牧和不可以放牧,可以放牧用1表示,否则用0表示,在这块牧场放牛,要求两个相邻的方格不能同时放牛,即牛与牛不能相 ...
- 【bzoj1688】[USACO2005 Open]Disease Manangement 疾病管理 状态压缩dp+背包dp
题目描述 Alas! A set of D (1 <= D <= 15) diseases (numbered 1..D) is running through the farm. Far ...
- POJ.3624 Charm Bracelet(DP 01背包)
POJ.3624 Charm Bracelet(DP 01背包) 题意分析 裸01背包 代码总览 #include <iostream> #include <cstdio> # ...
随机推荐
- EXCLE中快速插入图片
在excle中怎么快速插入图片呢,一张一张点实在比较麻烦 解决办法: <table><img src="D:\A.png" width="60" ...
- hdu 4857 逃生
题目连接 http://acm.hdu.edu.cn/showproblem.php?pid=4857 逃生 Description 糟糕的事情发生啦,现在大家都忙着逃命.但是逃命的通道很窄,大家只能 ...
- 使用 Swift 制作一个新闻通知中心插件(2)
我们在第一部分的文章中详细讲解了创建一个通知中心插件的整体过程.我们成功的在通知中心里面显示了新闻列表.但是截止到目前,我们还不能从通知中心的列表中查看新闻的详细内容.在这次的教程中,我们就以上次的教 ...
- iOS8 蓝牙设备的重连接(retrieve) Swift实现
今天App写到了蓝牙重连的阶段,以前针对sdk 6.0写的代码,蓝牙设备的回复是通过 - (void)retrievePeripherals:(NSArray *)peripheralUUIDs 然后 ...
- linq里的select和selectmany操作
Select() 和 SelectMany() 的工作都是依据源值生成一个或多个结果值.Select() 为每个源值生成一个结果值.因此,总体结果是一个与源集合具有相同元素数目的集合.与之相反,Sel ...
- RTLviewer与TechnologyMapViewer的区别?
区别: 1.QUARTUS II 中往往要查看RTL Viewer,其实RTLview是编译后的结果,显示的图形都是调用标准单元的结果,这是和思维有关联的显示结果,跟工艺库,FPGA类型,都没有关系: ...
- OpenGL学习笔记之了解OpenGL
OpenGL(全写Open GraphicsLibrary)是个定义了一个跨编程语言.跨平台的编程接口规格的专业的图形程序接口.它用于三维图像(二维的亦可),是一个功能强大,调用方便的底层图形库. 说 ...
- android selector
android 选择器的使用 1.在drawable文件夹下面建一个xml文件,如item.xml,在eclipse中有selector这个选项 2.可以在布局文件.xml(配置android:lis ...
- 团队项目--“我爱淘”校园二手书店 NABC分析
本项目的特点之一:可查询功能 NABC分析: N(Need):方便校园里的学生查找自己需要的二手书籍,免了同学想买二手书还得跑到阿姨那里去看. A(Approach):将学生的信息和书籍的信息都存放在 ...
- SVG基本图形及clipPath;
利用SVG可以实现很多复杂的图形,SVG的功能开发者们已经开发许多,今天初识一下SVG的基本图形绘制, <svg viewbox="0,0,400,400" style=&q ...