news 2026/6/10 1:20:46

【动态规划】01背包与完全背包问题详解,LeetCode零钱兑换II秒解,轻松解力扣

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
【动态规划】01背包与完全背包问题详解,LeetCode零钱兑换II秒解,轻松解力扣



👨‍💻程序员三明治:个人主页

🔥 个人专栏: 《设计模式精解》 《重学数据结构》
🤞先做到 再看见!

目录

  • 01背包题目分析
  • 01背包解决方法
  • 完全背包题目分析
  • 完全背包解决方法
  • LeetCode 518.零钱兑换II
    • 思路
    • 代码实现

01背包题目分析

有n件物品和一个最多能背重量为w 的背包。第i件物品的重量是weight[i],得到的价值是value[i] 。每件物品只能用一次,求解将哪些物品装入背包里物品价值总和最大。

每一件物品其实只有两个状态,取或者不取,所以可以使用回溯法搜索出所有的情况,那么时间复杂度就是O(2^n),这里的n表示物品数量。

所以暴力的解法是指数级别的时间复杂度。进而才需要动态规划的解法来进行优化!

在下面的讲解,我举一个例子:

物品为:

重量价值
物品0115
物品1320
物品2430

01背包解决方法

递归五部曲:

  1. 确定dp数组以及下标的含义:dp[i][j] 表示从下标为[0-i]的物品里任意取,放进容量为j的背包,价值总和最大是多少

为什么需要用二维数组呢?因为有两个维度需要分别表示:物品 和 背包容量

我们先看把物品0 放入背包的情况:

再看把物品1 放入背包:

背包容量为 0,放不下物品0 或者物品1,此时背包里的价值为0。

背包容量为 1,只能放下物品0,背包里的价值为15。

背包容量为 2,只能放下物品0,背包里的价值为15。

背包容量为 3,上一行同一状态,背包只能放物品0,这次也可以选择物品1了,背包可以放物品1 或者 物品0,物品1价值更大,背包里的价值为20。

背包容量为 4,上一行同一状态,背包只能放物品0,这次也可以选择物品1了,背包可以放下物品0 和 物品1,背包价值为35。

  1. 确定递推公式

对于递推公式,首先我们要明确有哪些方向可以推导出 dp[i][j]

这里我们dp[1][4]的状态来举例:

求取 dp[1][4] 有两种情况:

  1. 放物品1
  2. 还是不放物品1

如果不放物品1, 那么背包的价值应该是 dp[0][4] 即 容量为4的背包,只放物品0的情况。

如果放物品1,那么背包要先留出物品1的容量,目前容量是4,物品1 的容量(就是物品1的重量)为3,此时背包剩下容量为1。

容量为1,只考虑放物品0 的最大价值是 dp[0][1],这个值我们之前就计算过。

所以 放物品1 的情况 = dp[0][1] + 物品1 的价值,推导方向如图:

所以两种情况综合一起可以得出:

递归公式:<font style="color:rgb(71, 101, 130);background-color:rgba(27, 31, 35, 0.05);">dp[i][j] = max(dp[i - 1][j], dp[i - 1][j - weight[i]] + value[i]);</font>

3. dp数组初始化

首先从dp[i][j]的定义出发,如果背包容量j为0的话,即dp[i][0],无论是选取哪些物品,背包价值总和一定为0。如图:

从递归公式可以看出i 是由 i-1 推导出来,那么i为0的时候就一定要初始化。

  1. 确定遍历顺序

先遍历 物品还是先遍历背包重量呢?

其实都可以!! 但是先遍历物品更好理解

那么我先给出先遍历物品,然后遍历背包重量的代码。

for(inti=1;i<n;i++){for(intj=0;j<=bagweight;j++){if(j<weight[i]){dp[i][j]=dp[i-1][j];}else{dp[i][j]=Math.max(dp[i-1][j],dp[i-1][j-weight[i]]+value[i]);}}}
  1. 举例推导dp数组

最终结果就是dp[2][4]。

完全背包题目分析

完全背包和01背包问题唯一不同的地方就是,每种物品有无限件

在下面的讲解,继续用之前的例子:

物品为:

重量价值
物品0115
物品1320
物品2430

完全背包解决方法

  1. 确定dp数组以及下标的含义

dp[i][j] 表示从下标为[0-i]的物品,每个物品可以取无限次,放进容量为j的背包,价值总和最大是多少

  1. 确定递推公式

这里依然拿dp[1][4]的状态来举例

求取 dp[1][4] 有两种情况:

  1. 放物品1
  2. 还是不放物品1

如果不放物品1, 那么背包的价值应该是 dp[0][4] 即 容量为4的背包,只放物品0的情况。

如果放物品1,那么背包要先留出物品1的容量,目前容量是4,物品1 的容量(就是物品1的重量)为3,此时背包剩下容量为1。

容量为1,只考虑放物品0 和物品1 的最大价值是 dp[1][1],注意 这里和01背包有所不同了

在 01背包中,背包先空留出物品1的容量,此时容量为1,只考虑放物品0的最大价值是 dp[0][1],因为01背包每个物品只有一个,既然空出物品1,那背包中也不会再有物品1

而在完全背包中,物品是可以放无限个,所以 即使空出物品1空间重量,那背包中也可能还有物品1,所以此时我们依然考虑放 物品0 和 物品1 的最大价值即:dp[1][1], 而不是 dp[0][1]

所以可以得出

递推公式:<font style="color:rgb(71, 101, 130);background-color:rgba(27, 31, 35, 0.05);">dp[i][j] = max(dp[i - 1][j], dp[i][j - weight[i]] + value[i]);</font>

  1. dp数组初始化

如果背包容量j为0的话,即dp[i][0],无论是选取哪些物品,背包价值总和一定为0

再看其他情况:<font style="color:rgb(71, 101, 130);background-color:rgba(27, 31, 35, 0.05);">dp[i][j] = max(dp[i - 1][j], dp[i][j - weight[i]] + value[i]);</font>可以看出有一个方向 i 是由 i-1 推导出来,那么i为0的时候就一定要初始化。

dp[0][j],即:存放物品0的时候,各个容量的背包所能存放的最大价值。

for(intj=weight[0];j<=bagWeight;j++){dp[0][j]=dp[0][j-weight[0]]+value[0];}

  1. 确定遍历顺序

01背包二维DP数组,先遍历物品还是先遍历背包都是可以的。

因为两种遍历顺序,对于二维dp数组来说,递推公式所需要的值,二维dp数组里对应的位置都有。

  1. 举例推导dp数组

LeetCode 518.零钱兑换II

给你一个整数数组coins表示不同面额的硬币,另给一个整数amount表示总金额。

请你计算并返回可以凑成总金额的硬币组合数。如果任何硬币组合都无法凑出总金额,返回0

假设每一种面额的硬币有无限个。

题目数据保证结果符合 32 位带符号整数。

示例 1:

输入:amount = 5, coins = [1, 2, 5] 输出:4 解释:有四种方式可以凑成总金额: 5=5 5=2+2+1 5=2+1+1+1 5=1+1+1+1+1

示例 2:

输入:amount = 3, coins = [2] 输出:0 解释:只用面额 2 的硬币不能凑成总金额 3 。

思路

本题求的是装满这个背包的物品组合数是多少。因为每一种面额的硬币有无限个,所以这是完全背包

动规五部曲

  1. 确定dp数组以及下标的含义:

dp[i][j]:使用 下标为[0, i]的coins[i]能够凑满j(包括j)这么大容量的包,有dp[i][j]种组合方法。

  1. 递推公式

因为本题属于完全背包问题,所以递推公式需要参考

dp[i][j] = max(dp[i - 1][j], dp[i][j - weight[i]] + value[i])

但考虑到我们的dp数组含义求的是组合个数,所以本题的递推公式是

dp[i][j] = dp[i - 1][j] + dp[i - 1][j - nums[i]]

  1. 初始化

以这个为例

第一行如何初始化?

dp[0][0] 应该是多少?

背包空间为0,装满「物品0」 的组合数有多少呢?应该是 0 个, 但如果 「物品0」 的 数值就是0呢? 岂不是可以有无限个0 组合 和为0!

题目描述中说了<font style="color:rgb(71, 101, 130);background-color:rgba(27, 31, 35, 0.05);">1 <= coins.length <= 300</font><font style="color:rgba(38, 38, 38, 0.75);background-color:rgba(0, 10, 32, 0.03);">1 <= coins[i] <= 5000</font>,所以不用考虑 物品数值为0的情况。

dp[0][j]的含义:用「物品0」(即coins[0]) 装满 背包容量为j的背包,有几种组合方法。

如果 j 可以整除 物品0,那么装满背包就有1种组合方法。

for(intj=0;j<=amount;j++){if(j%coins[0]==0)dp[0][j]=1;}

最左列如何初始化呢?

dp[i][0] 的含义:用物品i(即coins[i]) 装满容量为0的背包 有几种组合方法。

都有一种方法,即不装。

所以 dp[i][0] 都初始化为1

综上,可以得出下图:

  1. 确定遍历顺序

先遍历物品,在遍历背包比较容易理解

  1. 打印dp数组

代码实现

classSolution{publicintchange(intamount,int[]coins){int[][]dp=newint[coins.length][amount+1];// 初始化最左列for(inti=0;i<coins.length;i++){dp[i][0]=1;}// 初始化最上行for(intj=0;j<=amount;j++){if(j%coins[0]==0)dp[0][j]=1;}// 开始遍历for(inti=1;i<coins.length;i++){for(intj=0;j<=amount;j++){if(coins[i]>j){dp[i][j]=dp[i-1][j];}else{dp[i][j]=dp[i-1][j]+dp[i][j-coins[i]];}}}returndp[coins.length-1][amount];}}




如果我的内容对你有帮助,请辛苦动动您的手指为我点赞,评论,收藏。感谢大家!!

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/8 21:55:47

错过Open-AutoGLM就等于错过未来医疗入口:你还在用传统方式挂号?

第一章&#xff1a;错过Open-AutoGLM就等于错过未来医疗入口在人工智能与医疗深度融合的今天&#xff0c;Open-AutoGLM 正成为推动智慧医疗跃迁的核心引擎。它不仅是一个开源的大语言模型框架&#xff0c;更是一套面向医疗场景深度优化的自动化推理系统&#xff0c;能够实现病历…

作者头像 李华
网站建设 2026/6/6 12:49:11

Open-AutoGLM核心算法解析,掌握NLP在会议纪要中的4大应用场景

第一章&#xff1a;Open-AutoGLM会议纪要生成在现代团队协作中&#xff0c;高效生成会议纪要成为提升沟通效率的关键环节。Open-AutoGLM 是一个基于开源大语言模型的自动化会议纪要生成系统&#xff0c;能够对接主流音视频会议平台&#xff0c;实时转录并提炼会议核心内容。系统…

作者头像 李华
网站建设 2026/6/9 8:21:19

从手动提醒到全自动预警:Open-AutoGLM保险到期管理的4次技术跃迁

第一章&#xff1a;从手动提醒到全自动预警&#xff1a;Open-AutoGLM保险到期管理的演进之路在保险资产管理领域&#xff0c;保单到期提醒曾长期依赖人工台账与邮件通知&#xff0c;效率低且易出错。随着系统规模扩大&#xff0c;传统方式难以应对高频、多维度的监控需求。Open…

作者头像 李华
网站建设 2026/6/9 23:16:14

为什么你的任务总不同步?Open-AutoGLM同步失败的8个常见陷阱

第一章&#xff1a;Open-AutoGLM同步失败的根源剖析在大规模语言模型自动化部署场景中&#xff0c;Open-AutoGLM作为核心调度组件&#xff0c;其同步机制的稳定性直接影响系统整体可用性。当出现同步失败时&#xff0c;通常源于配置、网络或权限三类根本原因。配置文件校验缺失…

作者头像 李华
网站建设 2026/6/5 17:57:35

【AI办公革命】:Open-AutoGLM如何实现会议纪要零手动输入?

第一章&#xff1a;Open-AutoGLM 会议纪要生成Open-AutoGLM 是一个基于开源大语言模型的自动化会议纪要生成系统&#xff0c;专为提升会议信息提取与结构化处理效率而设计。该系统融合语音识别、自然语言理解与文本摘要技术&#xff0c;能够从多源会议记录中自动生成清晰、准确…

作者头像 李华
网站建设 2026/6/9 19:49:15

从0到上线:中小企业如何用Open-AutoGLM搭建专属证件照服务平台

第一章&#xff1a;从0到上线&#xff1a;中小企业如何用Open-AutoGLM搭建专属证件照服务平台在数字化办公与远程服务快速发展的背景下&#xff0c;中小企业亟需低成本、高效率的自动化工具来提升服务能力。Open-AutoGLM 作为一款开源的多模态生成与推理框架&#xff0c;结合了…

作者头像 李华