题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2602

这是做的第一道01背包的题目。题目的大意是有n个物品,体积为v的背包。不断的放入物品,当然物品有各自的体积和价值。在不超过总体积v的情况下,问能够达到的最大价值。并且物品是一个一个放入的。最后若有剩余的体积也不会填满。

刚开始是用贪心做的。将价值与体积的比值设定为一个值,即单位价值。然后按照单位价值排序,挨个取物品,考虑到了体积为0的情况,就将单位价值设定为无穷大。但是这样做并不能保证最优解。例如:总体积为5,一共有两个物品。第一个体积4,价值6。第二个体积为2,价值为4。很明显第一个更合适,但是按照单位价值比来做。结果就不是这样的了。所以贪心法不适合这样的01背包。但是适合物品可以只取一部分的01背包,是的最后的背包肯定会被刚好填满。

在这道题中用数组dp来保存运算结果。例如dp[i][j]表示前i个物品放入体积为j的背包中。

判断第i个物品能否放入大小为j的背包中。如果可以,是放?还是不放?
重点是可以放的情况!

最后需要注意的是体积可以为0,同时最后结果保存在dp[n][v]中。

附上源代码:

 #include<iostream>
#include<stdio.h>
#include<stdlib.h>
#include<string>
#include<string.h>
#include<math.h>
#include<map>
#include<vector>
#include<algorithm>
using namespace std;
#define MAX 0x3f3f3f3f
#define MIN -0x3f3f3f3f
#define N 1005
int val[N];
int vol[N];
int dp[N][N];
int main()
{
int T;
int i, j;
int n, v;//物品,背包体积
scanf("%d", &T);
while (T--)
{
scanf("%d%d", &n, &v);
for (i = ; i <= n; i++)
scanf("%d", &val[i]);
for (i = ; i <= n; i++)
scanf("%d", &vol[i]);
memset(dp, , sizeof(dp));
for (i = ; i <= n; i++)
{
for (j = ; j <= v; j++)//体积可以是0
{
if (vol[i] <= j)//第i个物品的体积可以放到大小为j的背包中
dp[i][j] = max(dp[i - ][j], dp[i - ][j - vol[i]] + val[i]);//如果不放,前i-1件物品放入大小为v的包中。放的话,前i-1件物品放入v-vol[i]大的背包中
else
dp[i][j] = dp[i - ][j];//放不进去的话,相当于前i-1件物品放入大小为v的背包中
}
}
printf("%d\n", dp[n][v]);
}
return ;
}
顺便在附上错误的贪心代码,警告自己!
 #include<iostream>
#include<stdio.h>
#include<stdlib.h>
#include<string>
#include<string.h>
#include<math.h>
#include<map>
#include<vector>
#include<algorithm>
using namespace std;
#define MAX 0x3f3f3f3f
#define MIN -0x3f3f3f3f
struct node
{
int volume;
int vaule;
double f;
}ans[];
int cmp(const void *a, const void *b)
{
struct node *aa = (node *)a;
struct node *bb = (node *)b;
return(((aa->f)<(bb->f)) ? : -);
}
int main()
{
int T;
int N, V;
int i;
int val;//价值
int vol;//体积
scanf("%d", &T);
while (T--)
{
scanf("%d%d", &N, &V);
for (i = ; i < N; i++)
{
scanf("%d", &ans[i].vaule);
}
for (i = ; i < N; i++)
{
scanf("%d", &ans[i].volume);
if (ans[i].volume == )
ans[i].f = MAX;
else
ans[i].f = ans[i].vaule*1.0 / ans[i].volume;
}
qsort(ans, N, sizeof(ans[]), cmp);
val = ;
vol = ;
ans[N].vaule = ;
ans[N].volume = ;
ans[N].f = ;
for (i = ; i <= N; i++)
{
if (vol <= V)
{
val = val + ans[i].vaule;
vol = vol + ans[i].volume;
}
else if (vol>V)
{
val = val - ans[i - ].vaule;
//vol = vol - ans[i - 1].volume;
//val = val + (V - vol)*(int)ans[i - 1].f;
break;
}
}
printf("%d\n", val);
}
return ;
}

ACM HDU Bone Collector 01背包的更多相关文章

  1. HDU 2602 Bone Collector(01背包裸题)

    Bone Collector Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) T ...

  2. HDU 2602 - Bone Collector - [01背包模板题]

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2602 Many years ago , in Teddy’s hometown there was a ...

  3. HDU 2602 Bone Collector --01背包

    这种01背包的裸题,本来是不想写解题报告的.但是鉴于还没写过背包的解题报告.于是来一发. 这个真的是裸的01背包. 代码: #include <iostream> #include < ...

  4. [HDU 2602]Bone Collector ( 0-1背包水题 )

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2602 水题啊水题 还给我WA了好多次 因为我在j<w[i]的时候状态没有下传.. #includ ...

  5. HDU 2602 Bone Collector (01背包DP)

    题意:给定一个体积,和一些物品的价值和体积,问你最大的价值. 析:最基础的01背包,dp[i] 表示体积 i 时最大价值. 代码如下: #pragma comment(linker, "/S ...

  6. Bone Collector(01背包+记忆化搜索)

    Bone Collector Time Limit : 2000/1000ms (Java/Other)   Memory Limit : 32768/32768K (Java/Other) Tota ...

  7. hdoj 2620 Bone Collector(0-1背包)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2602 思路分析:该问题为经典的0-1背包问题:假设状态dp[i][v]表示前i件物品恰放入一个容量为v ...

  8. hdu2602 Bone Collector 01背包

    Many years ago , in Teddy’s hometown there was a man who was called “Bone Collector”. This man like ...

  9. [原]hdu2602 Bone Collector (01背包)

    本文出自:http://blog.csdn.net/svitter 题意:典型到不能再典型的01背包.给了我一遍AC的快感. //=================================== ...

随机推荐

  1. 做直线不要使用hr

    需要横线.竖线时不要使用hr标签来做,容易出问题,可以使用高度或宽度为1px并加上背景色的div.p代替.

  2. LeetCode 566. Reshape the Matrix (重塑矩阵)

    In MATLAB, there is a very useful function called 'reshape', which can reshape a matrix into a new o ...

  3. celery rabbit mq 详解

    Celery介绍和基本使用 Celery 是一个 基于python开发的分布式异步消息任务队列,通过它可以轻松的实现任务的异步处理, 如果你的业务场景中需要用到异步任务,就可以考虑使用celery, ...

  4. js判断是否使用的是微信浏览器

    代码如下: function is_weixin() { var ua = navigator.userAgent.toLowerCase(); return ua.match(/MicroMesse ...

  5. 从json_encode过来的的字符串被返回到html页面时的解析

    在工作过程中经常需要向服务器请求数据.在需要返回多个值的时候,使用json_encode处理数组然后返回是很常用的做法.如果没有指定返回数据类型的情况下,默认返回的是json格式的字符串.那么需要将这 ...

  6. NFS启动时报错Linux NFS:could not open connection for tcp6

    1.1 启动时出现的错误 [root@znix ~]#/etc/init.d/nfs start Shutting down NFS daemon:                          ...

  7. kettle介绍

    Kettle也叫PDI,在2006年Kettle加入了开源的BI组织Pentaho,正式命名为PDI,英文全称为Pentaho Data Integeration.Kettle是"Kettl ...

  8. MapReduce简单分析

    在Map端 数据从Map中写入环形缓冲区,进行分区,分区时达到80%后溢出写入到磁盘,这几步同步进行 中间有个Shuffle过程 Reduce端 执行完Map 后到Reduce内存中,进行sort和m ...

  9. arguments对象

    改变函数arguments也会改变对应的参数,arguments会自动映射到对应的参数上 但是改变参数并不会同步改变arguments 它们并不公用内存 在strict模式,不能对arguments付 ...

  10. [转载] 应用于负载均衡的一致性哈希及java实现

    转载自http://blog.csdn.net/haitao111313/article/details/7537799 这几天看了几遍一致性哈希的文章,但是都没有比较完整的实现,因此试着实现了一下, ...