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的火车运输,问最少需要运送多少次可以将货物运完 思 ...
随机推荐
- Elasticsearch中提升大文件检索性能的一些总结
笔者在实际生产环境中经常遇到一些大文件的检索,例如一些书籍内容,PDF文件等.今天这篇博客主要来探讨下如何提升ES在检索大文件的一些性能,经验有限,算是一个小小的总结吧! 1.大文件是多大? E ...
- 软件工程作业——Word Counter
github地址 https://github.com/Pryriat/Word_Counter 项目说明 wc.exe 是一个常见的工具,它能统计文本文件的字符数.单词数和行数.这个项目要求写一个命 ...
- 如何選擇最佳的 Wi-Fi 無線網路頻道,獲得最佳的傳輸速度(转载)
转自:https://blog.gtwang.org/useful-tools/how-to-find-the-best-wi-fi-channel-for-your-router/
- CentOS 6.5 安装Clang 3.5.0
来自引用: http://www.cnblogs.com/dudu/p/4294374.html 编译llvm几乎耗费了1个小时-.. 编译CoreCLR需要Clang 3.5,而CentOS上安装的 ...
- 洛谷P1031 均分纸牌
P1031 均分纸牌 题目描述 有 N 堆纸牌,编号分别为 1,2,…, N.每堆上有若干张,但纸牌总数必为 N 的倍数.可以在任一堆上取若干张纸牌,然后移动. 移牌规则为:在编号为 1 堆上取的纸牌 ...
- 洛谷P1582 倒水
P1582 倒水 题目描述 一天,CC买了N个容量可以认为是无限大的瓶子,开始时每个瓶子里有1升水.接着~~CC发现瓶子实在太多了,于是他决定保留不超过K个瓶子.每次他选择两个当前含水量相同的瓶子,把 ...
- 1.函数的结构,调用,传参,形参,实参,args,kwargs,名称空间,高阶函数
1.函数的初识 初始函数 获取任意一个字符串的元素的个数 s1='dsjdkjkfefenga' count=0 for i in s1: count+=1 print(count) 获取列表的元素的 ...
- jsp页面包含的几中方式
(1)include指令 include指令告诉容器:复制被包含文件汇总的所有内容,再把它粘贴到这个文件中. <%@ include file="Header.jsp"%&g ...
- NET Core 2.1.0 now available
ASP.NET Core 2.1.0 now available https://blogs.msdn.microsoft.com/webdev/2018/05/30/asp-net-core-2-1 ...
- MDX属性查询
SELECT NON EMPTY { { { { { AddCalculatedMembers([会员.会员ID].[会员ID].Members), [会员.会员ID].[(All)] } } } } ...