四种常见背包问题包括:① 最优装配 ② 部分背包问题 ③ 01背包问题 ④ 完全背包问题
给出 n 个物体,重量分别为 wi,使总重量不超过容量 C 的情况下选择尽量多的物体。
这种背包问题很简单,只要求数量多就可以,因此我们对重量进行排序贪心选择就好。
参考代码:
#include
#include
using namespace std;
int *stuff;
// 最优装配
int getResult(int n, int c){
int cnt = 0;
int sum = 0;
for(int i = 0; i < n; i++){
sum += stuff[i];
if(sum <= c) cnt++;
else break;
}
return cnt;
}
int main(){
int n, c;
while(cin >> n >> c){
stuff = new int[n];
for(int i = 0; i < n; i++){
cin >> stuff[i];
}
sort(stuff, stuff+n);
int cnt = getResult(n, c);
cout << cnt << endl;
delete[] stuff;
}
return 0;
}
有 n 个物体,重量分别为 wi,价值为 vi,在总重量不超过容量 C 的情况下让总价值最高,
每个物体可以只取走一部分,若取走部分物体则价值也会等比例减少。
这种背包问题也比较简单,可以定义一个变量 rate 表示各个物体的性价比,
性价比最高的物体先拿走,如果可以整个物体都拿走则直接拿走,若不能整个拿走就拿走部分。
参考代码:
#include
#include
using namespace std;
struct STUFF {
int weight;
int price;
double rate; // 性价比
bool operator < (STUFF other) const {
return rate > other.rate;
}
}*stuffs;
// 部分背包问题
double getResult(int n, int c){
double sum = 0;
for(int i = 0; i < n; i++){
// 可以整个拿走就直接拿走
if(c >= stuffs[i].weight){
sum += stuffs[i].price;
c -= stuffs[i].weight;
}
else{
// 不能直接拿走就拿部分
sum += stuffs[i].price*1.0*c/stuffs[i].weight;
break;
}
}
return sum;
}
int main(){
int n, c;
while(cin >> n >> c){
stuffs = new STUFF[n];
for(int i = 0; i < n; i++){
cin >> stuffs[i].weight >> stuffs[i].price;
stuffs[i].rate = stuffs[i].price*1.0/stuffs[i].weight;
}
sort(stuffs, stuffs+n);
double result = getResult(n, c);
cout << result << endl;
delete[] stuffs;
}
return 0;
}
有 n 个物体,重量分别为 wi,价值为 vi,在总重量不超过容量 C 的情况下让总价值最高,
但不允许只取走部分物体,要拿走只能整个拿走。
从01背包开始就涉及到了动态规划的知识,动态规划的本质就是递推。
先打表进行观察:
表中 dp[ i ] [ j ] 表示容量为 j,可以在 0 号物品到 i 号物品中进行任意组合,这种情况下的最大值。
第一行与第一列都不用说,一眼就可以直接得到答案。
剩下的部分怎么进行选择呢?
比如 dp[2] [4] = 6,我们已经知道了 dp[1] [4] = 5,说明在没有考虑 2 号物品的时候最大价值为 5。
现在加入了 2 号物品需要考虑,(3,4)重量为 3,价值为 4。
那么必须要选择 2 号物品才有可能超过之前计算不考虑 2 号物品的最大价值 5。
而选择 2 号物品的最大价值为多少呢?
应该是 dp[2-1] [4-3] + 4 = dp[1] [1] + 4 = 2 + 4 = 6,
dp[1] [1] 为不考虑 2 号物品,容量为 1 的最大价值,这样正好可以装下重量为 3 的 2 号物品且价值最大。
因此我们只要比较加入 2 号物品的最大价值与不加入 2 号物品的最大价值即可,
即 max(dp[1] [1] + 4, dp[1] [4])。
一般情况:dp[i] [j] = max( dp[i-1] [j-w] + v, dp[i-1] [j] )。
所以处理到最后 dp[2] [5] 就是我们要求的答案。
参考代码:
#include
#include
using namespace std;
int **stuffs; // 物品数组
int **dp; // 二维数组保存dp表
void show(int **arr, int n, int c){
for(int i = 0; i < n; i++){
for(int j = 0; j <= c; j++){
cout << arr[i][j] << " ";
}
cout << endl;
}
}
// 得到dp表
int getDpTable(int n, int c){
// 初始化第一行
for(int i = 0; i < c+1; i++){
if(i >= stuffs[0][0]){
dp[0][i] = stuffs[0][1];
}
else{
dp[0][i] = 0;
}
}
// 初始化第一列
for(int i = 0; i < n; i++){
dp[i][0] = 0;
}
// currentWeight和currentValue保存当前物品的重量与价值
int currentWeight, currentValue;
for(int i = 1; i < n; i++){
currentWeight = stuffs[i][0];
currentValue = stuffs[i][1];
for(int j = 1; j < c+1; j++){ // 注意i和j都是从1开始,j表示背包的容量
if(j >= currentWeight){
int value1 = currentValue+dp[i-1][j-currentWeight]; // 该物品的价值 + 没有加上该物品时的最大价值
int value2 = dp[i-1][j]; // 使用前面i个物品组合成的最大价值
dp[i][j] = max(value1, value2);
}
else{
dp[i][j] = dp[i-1][j];
}
}
}
show(dp, n, c);
return dp[n-1][c];
}
int main(){
int n, c;
while(cin >> n >> c){
stuffs = new int*[n];
dp = new int*[n];
for(int i = 0; i < n; i++){
stuffs[i] = new int[2];
dp[i] = new int[c+1];
}
for(int i = 0; i < n; i++){
cin >> stuffs[i][0] >> stuffs[i][1];
}
cout << getDpTable(n, c) << endl;
for(int i = 0; i < n; i++){
delete[] stuffs[i];
delete[] dp[i];
}
delete[] stuffs;
delete[] dp;
}
return 0;
}
在01背包的基础上添加了新的条件:各个物品不再是一个,而是变成一种,可以拿多个。
继续打表观察:
前面三种情况都很好分析,重点是最后一种情况。
这里的 v[i]+dp[i] [j-w[i]] 注意是 i 而不是01背包中的 i-1,是同一行。
虽然在完全背包中一个物品可以选择多个,但是并不需要通过 for 循环来确定选择 k 个 i 物品时价值最大。
我们只要简单分成两个状态就足够了: 1. 选择 i 物品 2. 不选择 i 物品
而 dp[i-1] [j] 就是不选择 i 物品时的最大价值 maxValue。
v[i] + dp[i] [j-w[i]] 是选择 i 物品时的最大价值maxValue,这里 v[i] 表示选择一个 i 物品,
但是选择一个 i 物品不保证是最大价值啊,这里就要看 dp[i] [j-w[i]] 了。
dp[i] [j-w[i]] 就已经包含了 k-1 个 i 物品的最大价值,因此两者组合起来就是 k 个 i 物品使总价值最大。
(01背包只能选一个,因此为 v[i] + dp[i-1] [j-w[i]],
完全背包问题可以选多个,因此为 v[i] + dp[i] [j-w[i]])
参考代码:
#include
#include
using namespace std;
struct STUFF {
int weight;
int value;
}*stuffs;
int **dp;
int getResult(int n, int c){
for(int i = 1; i <= c; i++){
dp[0][i] = (i/stuffs[0].weight)*stuffs[0].value; // 初始化第一行 只能取第一个物品
}
for(int i = 1; i < n; i++){
for(int j = 1; j <= c; j++){
if(stuffs[i].weight > j){
dp[i][j] = dp[i-1][j]; // 如果买不起该物品 最大价值就是不买该物品时计算出来的最大值
}
else{
// 两种情况:1.买该物品 2.不买该物品
// dp[i-1][j]就是不买该物品的最大价值
// value[i]+dp[i][j-weight[i]]是买该物品时的最大价值 i物品被这里被选择了k次而不是1次 注意该重点
dp[i][j] = max(dp[i-1][j], stuffs[i].value+dp[i][j-stuffs[i].weight]);
}
}
}
for(int i = 0; i < n; i++){
for(int j = 0; j <= c; j++){
cout << dp[i][j] << " ";
}
cout << endl;
}
return dp[n-1][c];
}
int main(){
int n, c;
while(cin >> n >> c){
stuffs = new STUFF[n];
dp = new int*[n];
for(int i = 0; i < n; i++){
dp[i] = new int[c+1]; // 注意是c+1 dp数组中有容量为0的情况
}
for(int i = 0; i < n; i++){
for(int j = 0; j <= c; j++){
dp[i][j] = 0;
}
}
for(int i = 0; i < n; i++){
cin >> stuffs[i].weight >> stuffs[i].value;
}
int ans = getResult(n, c);
cout << ans << endl;
for(int i = 0; i < n; i++){
delete[] dp[i];
}
delete[] dp;
delete[] stuffs;
}
return 0;
}
【END】感谢观看