动态规划——完全背包问题

目录

一、完全背包

二、零钱兑换

三、零钱兑换II

四、完全平方数


一、完全背包

完全背包可以描述为:有n件物品和一个最多能背体积为 v 的背包。第 i 件物品的体积是 v[i],得到的价值是value[i] 。每件物品有无限多个,求解将哪些物品装入背包里物品价值总和最大。

在完全背包问题中,对于每件物品,有多种状态:不选,选一个,选两个......

题目

完全背包

题目解析

第一问的所有情况就是上图红色方框的内容, 第二问的所有情况就是上图蓝色方框的内容。

解决第一问

第一步:确定状态表示

dp[i][j]:表示从前 i 个物品中挑选,总体积不超过 j 的所有选法中,能挑选出来的物品价值最大值。

第二步:推出状态转移方程

前面我们已经说明了,完全背包中的每件物品的数量有无限个,所以对于每件物品来说,有选或者选1个,选2个...... 多种状态。根据最后一步的状况,分情况讨论,即要么不选择 i 号物品,要么选 1个 i 号物品,要么选 2个 i 号物品,要么选 3个 i 号物品......

选一个 i 物品,这时就有一个v[i]的体积了,然后仅需去 1 ~ i - 1 区间选不超过 j - v[i] 的最大价值,然后再加上 i 的价值。

选两个 i 物品,这时就有两个v[i]的体积了,然后仅需去 1 ~ i - 1 区间选不超过 j - 2*v[i] 的最大价值,然后再加上两个 i 的价值。

选n个 i 物品,这时就有n个v[i]的体积了,然后仅需去 1 ~ i - 1 区间选不超过 j - n*v[i] 的最大价值,然后再加上n个 i 的价值。

第三步:初始化dp表

第一列不需要初始化。

第一行表示没有物品,当 j 为0的时候,要凑成不超过 j,只要不选就行了,最大价值是0。如果j为1、2、3......但是没有物品可选,所以最大价值也是0。

解决第二问

第一步:确定状态表示

dp[i][j]:表示从前 i 个物品中挑选,总体积恰好为 j 的所有选法中,能挑选出来的物品价值最大值。

第二步:推出状态转移方程

状态转移方程的推导方法和第一问相同,只是有一些细节问题需要注意。

题目要求选出的物品中,总体积要恰好为 j ,那么就有可能怎么选也凑不出物品的总体积恰好为 j 的情况,那么我们就不能使用这种情况,可以用 dp[i][j] == -1 来表示凑不出物品的总体积恰好为 j 的情况,即这种情况不存在。

第三步:初始化dp表

第一列不需要初始化。

第一行表示没有物品,当 j 为0的时候,要凑成不超过 j,只要不选就行了,最大价值是0。如果j为1、2、3......因为没有物品可选,所以这些情况都不存在,值为-1。

解题代码:

cpp 复制代码
#include <iostream>
#include <vector>
using namespace std;

int main() 
{
    int n, V;
    cin >> n >> V;
    vector<int> v(n);
    vector<int> w(n);
    for(int i = 0; i < n; i++)
        cin >> v[i] >> w[i];

    vector<vector<int>> dp1(n+1, vector<int>(V+1));
    for(int i = 1; i <= n; i++)
    {
        for(int j = 0; j <= V; j++)
        {
            dp1[i][j] = dp1[i-1][j];
            if(j >= v[i-1])
            {
                int ret = max(dp1[i-1][j], dp1[i][j-v[i-1]] + w[i-1]);
                dp1[i][j] = max(dp1[i][j], ret);
            }
        }
    }

    vector<vector<int>> dp2(n+1, vector<int>(V+1));
    for(int i = 1; i <= V; i++)
        dp2[0][i] = -1; // 用-1来表示该情况不存在
    for(int i = 1; i <= n; i++)
    {
        for(int j = 0; j <= V; j++)
        {
            dp2[i][j] = dp2[i-1][j];
            if(j >= v[i-1] && dp2[i][j-v[i-1]] != -1)
            {
                int ret = max(dp2[i-1][j], dp2[i][j-v[i-1]] + w[i-1]);
                dp2[i][j] = max(dp2[i][j], ret);
            }
        }
    }

    cout << dp1[n][V] << endl;
    cout << (dp2[n][V] == -1 ? 0 : dp2[n][V]) << endl;
    return 0;
}

二、零钱兑换

零钱兑换

第一步:确定状态表示

dp[i][j]:表示从前 i 个硬币中挑选,使其总和恰好等于j,所有选法中,最少的硬币个数。

第二步:推出状态转移方程

第三步:初始化dp表

第一列不用初始化。第一行表示没有硬币,当 j 为0表示硬币总和为0,不选就行了。

解题代码:

cpp 复制代码
class Solution 
{
public:
    int coinChange(vector<int>& coins, int amount) 
    {
        int m = coins.size();
        vector<vector<int>> dp(m+1, vector<int>(amount+1));
        for(int i = 1; i <= amount; i++)
            dp[0][i] = 0x3f3f3f3f;
        for(int i = 1; i <= m; i++)
        {
            for(int j = 0; j <= amount; j++)
            {
                dp[i][j] = dp[i-1][j];
                if(j >= coins[i-1])
                    dp[i][j] = min(dp[i][j], dp[i][j-coins[i-1]]+1);
            }
        }
        return dp[m][amount] == 0x3f3f3f3f ? -1 : dp[m][amount];
    }
};

三、零钱兑换II

零钱兑换II

第一步:确定状态表示

dp[i][j]:表示从前 i 个硬币中挑选,使其总和恰好等于j,一共有多少种选法。

第二步:推出状态转移方程

第三步:初始化dp表

第一列不用初始化。

第一行表示没有硬币,当 j = 0时表示总和为0,不选就行了,这是一种选法。当j = 1、2、3...,因为没有硬币,所以根本凑不出总和等于j的选法。直接都给0就行了。

解题代码:

cpp 复制代码
class Solution 
{
public:
    int change(int amount, vector<int>& coins) 
    {
        int m = coins.size();
        vector<vector<double>> dp(m+1, vector<double>(amount+1));
        dp[0][0] = 1;
        for(int i = 1; i <= m; i++)
        {
            for(int j = 0; j <= amount; j++)
            {
                dp[i][j] += dp[i-1][j];
                if(j >= coins[i-1])
                    dp[i][j] += dp[i][j-coins[i-1]];
            }
        }
        return dp[m][amount];
    }
};

四、完全平方数

完全平方数

第一步:确定状态表示

dp[i][j]:表示从前 i 个完全平方数种挑选,使其总和恰好等于 j,所有选法中,平方数的最少数量。

第二步:推出状态转移方程

第三步:初始化dp表

第一列不用初始化。

第一行表示完全平方数为0,当 j = 0表示总和为0,不选就行了,最少数量为0。当 j = 1、2...时完全平方数为0,根本凑不出和为 j,然后我们填dp[i][j]要最小值,为了不让这些位置得值影响填表,因此可以给0x3f3f3f3f。

解题代码:

cpp 复制代码
class Solution 
{
public:
    int numSquares(int n) 
    {
        int m = sqrt(n);
        vector<vector<int>> dp(m+1, vector<int>(n+1));
        for(int i = 1; i <= n; i++)
            dp[0][i] = 0x3f3f3f3f;
        for(int i = 1; i <= m; i++)
        {
            for(int j = 0; j <= n; j++)
            {
                dp[i][j] = dp[i-1][j];
                if(j >= i*i)
                    dp[i][j] = min(dp[i][j], dp[i][j-i*i]+1);
            }
        }
        return dp[m][n];
    }
};
相关推荐
Coovally AI模型快速验证38 分钟前
MMYOLO:打破单一模式限制,多模态目标检测的革命性突破!
人工智能·算法·yolo·目标检测·机器学习·计算机视觉·目标跟踪
可为测控1 小时前
图像处理基础(4):高斯滤波器详解
人工智能·算法·计算机视觉
Milk夜雨2 小时前
头歌实训作业 算法设计与分析-贪心算法(第3关:活动安排问题)
算法·贪心算法
BoBoo文睡不醒2 小时前
动态规划(DP)(细致讲解+例题分析)
算法·动态规划
apz_end3 小时前
埃氏算法C++实现: 快速输出质数( 素数 )
开发语言·c++·算法·埃氏算法
仟濹3 小时前
【贪心算法】洛谷P1106 - 删数问题
c语言·c++·算法·贪心算法
CM莫问4 小时前
python实战(十五)——中文手写体数字图像CNN分类
人工智能·python·深度学习·算法·cnn·图像分类·手写体识别
sz66cm4 小时前
LeetCode刷题 -- 45.跳跃游戏 II
算法·leetcode
Amor风信子4 小时前
华为OD机试真题---战场索敌
java·开发语言·算法·华为od·华为
old_power5 小时前
【PCL】Segmentation 模块—— 基于图割算法的点云分割(Min-Cut Based Segmentation)
c++·算法·计算机视觉·3d