LeetCode518. 零钱兑换 II 以及 动态规划相关的排列组合问题

文章目录

      • 一、题目
      • 二、题解
        • 方法一:完全背包问题的变体(版本1)
        • 方法二:完全背包问题变体(版本2)
      • 三、拓展:先遍历物品后遍历背包vs先遍历背包后遍历物品
        • 先遍历物品后遍历背包(组合问题)
        • 先遍历背包后遍历物品(排列问题)


一、题目

给你一个整数数组 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 。

示例 3:

输入:amount = 10, coins = [10] 
输出:1

提示:

  • 1 <= coins.length <= 300
  • 1 <= coins[i] <= 5000
  • coins 中的所有值 互不相同
  • 0 <= amount <= 5000

二、题解

方法一:完全背包问题的变体(版本1)

题目理解

这道题目是一个动态规划问题,需要计算凑成总金额的硬币组合数。给定一组硬币的面额数组 coins 和一个总金额 amount,要求计算有多少种不同的组合方式来凑成总金额。每个硬币的面额都可以被使用无限次。

动态规划思路

我们可以将硬币问题与完全背包问题联系起来:

  • 将硬币的面额视为物品的重量。
  • 将总金额视为背包的容量。
  • 将计算硬币组合数的问题视为在完全背包问题中计算组合数量的变种。
  1. 定义状态

我们需要定义一个状态来表示问题的子问题和最优解。在这个问题中,我们可以使用二维数组 dp[i][j] 来表示前 i 种硬币组成总金额 j 的组合数。其中,i 表示考虑的硬币种类数量,j 表示总金额。

  1. 初始化状态

我们需要初始化状态数组 dp,确保其初始值是正确的。在这里,可以看到 dp[i][0] 应该初始化为0,因为没有硬币可供选择。当 i % coins[0] == 0dp[0][i] 应该初始化为1,因为i可以由整数个第一个硬币组成。

  1. 状态转移方程

接下来,我们需要找到状态之间的转移关系,即如何从子问题的最优解推导出原问题的最优解。在这个问题中,状态转移方程如下:

  • 如果当前总金额 j 小于硬币面额 coins[i],则无法将硬币 i 加入组合,所以 dp[i][j] = dp[i-1][j],表示不使用硬币 i
  • 如果 j 大于等于硬币面额 coins[i],我们可以选择使用硬币 i 或者不使用。因此,dp[i][j] 等于两者之和:
    • 不使用硬币 i,即 dp[i-1][j]
    • 使用硬币 i,即 dp[i][j - coins[i]],这里的 dp[i][j - coins[i]] 表示在考虑硬币 i 时,总金额减去硬币 i 的面额后的组合数。
  1. 填充状态表格

通过上述状态转移方程,我们可以通过双重循环遍历所有的子问题,从而填充状态表格 dp。外层循环遍历硬币种类 i,内层循环遍历总金额 j,根据状态转移方程更新 dp[i][j]

  1. 获取最终答案

最后,我们可以通过 dp[coins.size() - 1][amount] 来获取问题的最终答案,即考虑了所有硬币种类并且总金额为 amount 时的组合数。

代码解析

class Solution {
public:int change(int amount, vector<int>& coins) {vector<vector<int>> dp(coins.size(), vector<int>(amount + 1, 0));for (int i = 0; i <= amount; i++) {if (i % coins[0] == 0) {dp[0][i] = 1;}}for (int i = 1; i < coins.size(); i++) {for (int j = 0; j <= amount; j++) {if (j < coins[i]) {dp[i][j] = dp[i - 1][j];} else {dp[i][j] = dp[i - 1][j] + dp[i][j - coins[i]];}}}return dp[coins.size() - 1][amount];}
};
  1. 创建一个二维数组 dp,其中 dp[i][j] 表示前 i 种硬币组成总金额 j 的组合数。

  2. 初始化 dp[0][j],即考虑只有一种硬币时,对应总金额 j 的组合数。如果 j 可以被第一种硬币整除,那么 dp[0][j] 初始化为1,表示有一种组合方式,即只使用第一种硬币。

  3. 通过嵌套的循环遍历硬币种类 i 和总金额 j,根据状态转移方程更新 dp[i][j]。如果 j 小于硬币面额 coins[i],则 dp[i][j] 等于 dp[i-1][j],否则 dp[i][j] 等于 dp[i-1][j] + dp[i][j - coins[i]]

  4. 最后返回 dp[coins.size() - 1][amount],即考虑了所有硬币种类并且总金额为 amount 时的组合数。

方法二:完全背包问题变体(版本2)

  1. 定义状态

首先,我们需要定义一个状态,来表示问题的子问题和最优解。在这个问题中,我们可以使用一维数组 dp,其中 dp[i] 表示总金额 i 的组合方式数量。

  1. 初始化状态

接下来,我们需要初始化状态数组 dp,确保其初始值是正确的。在这里,可以看到 dp[0] 应该初始化为1,因为总金额为0时,只有一种组合方式,那就是什么硬币都不选。

  1. 状态转移方程

然后,我们需要找到状态之间的转移关系,即如何从子问题的最优解推导出原问题的最优解。状态转移方程如下:

  • 对于每个硬币面额 coins[i],我们可以选择使用该硬币或不使用。
  • 如果我们选择使用硬币 coins[i],那么 dp[j] 应该等于 dp[j] + dp[j - coins[i]],表示在考虑硬币 coins[i] 时,总金额 j 的组合方式数量应该加上总金额 j - coins[i] 的组合方式数量。
  • 如果我们选择不使用硬币 coins[i],那么 dp[j] 保持不变。
  1. 填充状态数组

通过上述状态转移方程,我们可以通过循环遍历所有的子问题,从而填充状态数组 dp。外层循环遍历硬币的面额 i,内层循环遍历总金额 j,根据状态转移方程更新 dp[j]

  1. 获取最终答案

最后,我们可以通过 dp[amount] 来获取问题的最终答案,即总金额为 amount 时的组合方式数量。

代码解析

class Solution {
public:int change(int amount, vector<int>& coins) {vector<int> dp(amount+1,0);dp[0] = 1;for(int i = 0; i < coins.size(); i++){for(int j = coins[i]; j <= amount; j++){dp[j] += dp[j-coins[i]];}}return dp[amount];}
};
  1. 创建一个一维数组 dp,其中 dp[i] 表示总金额 i 的组合方式数量。

  2. 初始化 dp[0] 为1,因为总金额为0时,只有一种组合方式,即不选硬币。

  3. 通过嵌套的循环遍历硬币面额 coins[i] 和总金额 amount,根据状态转移方程 dp[j] += dp[j - coins[i]] 来更新 dp[j]。这表示在考虑硬币 coins[i] 时,总金额 j 的组合方式数量应该加上总金额 j - coins[i] 的组合方式数量。

  4. 最后返回 dp[amount],即总金额为 amount 时的组合方式数量。

二维数组版本

class Solution {
public:int change(int amount, vector<int>& coins) {vector<vector<int>> dp(coins.size(), vector<int>(amount + 1, 0));// 初始化第一行,当金额为0时,有1种方式,即不选择任何硬币for (int i = 0; i < coins.size(); i++) {dp[i][0] = 1;}for (int i = 0; i < coins.size(); i++) {for (int j = 1; j <= amount; j++) {// 如果当前硬币面值大于金额j,则不能选择当前硬币,直接继承上一种方式的数量if (coins[i] > j) {if (i > 0) {dp[i][j] = dp[i - 1][j];}} else {// 否则,可以选择当前硬币或不选择当前硬币if (i > 0) {dp[i][j] = dp[i - 1][j] + dp[i][j - coins[i]];} else {dp[i][j] = dp[i][j - coins[i]];}}}}return dp[coins.size() - 1][amount];}
};

三、拓展:先遍历物品后遍历背包vs先遍历背包后遍历物品

先遍历物品后遍历背包(组合问题)

如果我们选择先遍历物品后遍历背包,那么我们的状态 dp[j] 表示的是总金额为 j 时的硬币组合数量。在这种情况下,我们考虑了每个硬币,并决定是否将其放入组合。这导致了我们计算的是硬币的组合数量,而不考虑硬币的排列顺序。

#include <iostream>
#include <vector>
using namespace std;class Solution {
public:int change(int amount, vector<int>& coins) {vector<int> dp(amount + 1, 0); // 初始化动态规划数组,dp[i] 表示凑成金额 i 的方法数dp[0] = 1; // 凑成金额为 0 的方法数为 1,因为什么都不选也是一种方法for (int i = 0; i < coins.size(); i++) {int coin = coins[i];for (int j = coin; j <= amount; j++) {// 如果当前硬币面额小于等于当前金额 j,可以考虑将该硬币加入方案// 当前 dp[j] 的值应该加上 dp[j-coin],表示不使用这个硬币时的方法数dp[j] += dp[j - coin];}// 输出当前填写后的 dp 数组cout << "Coins: " << coin << " | DP Array: ";for (int k = 0; k <= amount; k++) {cout << dp[k] << " ";}cout << endl;}return dp[amount]; // 返回凑成目标金额的方法数}
};int main() {Solution s;vector<int> coins = { 1, 2, 5 };int amount = 5;int result = s.change(amount, coins);cout << "Result: " << result << endl;return 0;
}

在这里插入图片描述

[注意]大多数格子由三行组成,第一行是推导出结果的公式,第三行是推导出的结果,第二行是组合方式。

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

先遍历背包后遍历物品(排列问题)

如果我们选择先遍历背包后遍历物品,那么我们的状态 dp[j] 表示的是总金额为 j 时的硬币排列数量。在这种情况下,我们考虑了每个背包容量,然后决定放入哪些硬币。这导致了我们计算的是硬币的排列数量,考虑了硬币的顺序。

注意:我写了两个版本,一个是一维数组,一个是二维数组,最直观的就是展现成二维数组。

一维数组

#include <iostream>
using namespace std;
#include <vector>class Solution {
public:int change(int amount, vector<int>& coins) {vector<int> dp(amount + 1, 0);dp[0] = 1;for (int j = 0 ; j <= amount; j++) {for (int i = 0; i < coins.size(); i++) {if(j >= coins[i]) dp[j] += dp[j - coins[i]];}}return dp[amount];}
};int main()
{Solution s;vector<int> coins;coins.push_back(1); coins.push_back(2); coins.push_back(5);int result;result = s.change(5, coins);cout << "result:" << result << endl;
}

二维数组
下面这段代码中dp[j][i]的意义是背包为j大小时,最后一个放入价值为coins[i]硬币加上不放入该硬币(最后一个投入的硬币是从coins[0]coins[i]之间的硬币)的方法种类总数。

#include <iostream>
#include <vector>
using namespace std;class Solution {
public:int change(int amount, vector<int>& coins) {vector<vector<int>> dp(amount+1,vector<int>(coins.size(),0)); dp[0][0] = 1; int j = 0;for (j = 0; j <=amount ; j++) {for (int i = 0; i <coins.size(); i++) {if( i > 0 ) dp[j][i] = dp[j][i - 1];if (j >= coins[i]) {dp[j][i]+=dp[j - coins[i]][coins.size() - 1];}    }// 输出当前填写后的 dp 数组cout << "j(容量) " << j << " | DP Array: ";for (int k = 0; k < coins.size(); k++) {cout << dp[j][k] << " ";}cout << endl;}return dp[amount][coins.size()-1]; // 返回凑成目标金额的方法数}
};int main() {Solution s;vector<int> coins = { 1, 2, 5 };int amount = 5;int result = s.change(amount, coins);cout << "Result: " << result << endl;return 0;
}

在这里插入图片描述

[注意]大多数格子由三行组成,第一行是推导出结果的公式,第三行是推导出的结果,第二行是排列方式。


转置这个矩阵(如果看上面矩阵不顺眼)

#include <iostream>
#include <vector>
using namespace std;class Solution {
public:int change(int amount, vector<int>& coins) {vector<vector<int>> dp(coins.size(), vector<int>(amount + 1, 0));dp[0][0] = 1;int i = 0;for (i = 0; i <= amount; i++) {for (int j = 0; j < coins.size(); j++) {if (j > 0) dp[j][i] = dp[j-1][i];if (i >= coins[j]) {dp[j][i] += dp[coins.size()-1][i-coins[j]];}}}for (int i = 0; i < coins.size(); i++) {for (int j = 0; j <= amount; j++) {cout << dp[i][j] << " ";}cout << endl;}return dp[coins.size() - 1][amount]; // 返回凑成目标金额的方法数}
};int main() {Solution s;vector<int> coins = { 1, 2, 5 };int amount = 5;int result = s.change(amount, coins);cout << "Result: " << result << endl;return 0;
}

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/107809.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

高性能数据JS网格 Bryntum Grid 5.5.2 Crack

高性能数据网格 Bryntum Grid 是一个高性能的网络表格组件。它是用纯 JavaScript 构建的&#xff0c;并且可以轻松地与所有主要 JS 框架集成。 功能丰富 Bryntum Grid 具有您期望从专业网格组件获得的所有功能&#xff0c;包括&#xff1a; 很好的表现;很好的绩效 没有人喜欢缓…

记录我在cmd里使用pip命令下载Python的包时碰见的两个错误

1、pip时报错&#xff1a;Defaulting to user installation because normal site-packages is not writeable 解决方法&#xff1a;在 pip install 后面加上 --user 即可&#xff0c;这个是权限不足引发的问题。如果还是不行则用镜像源&#xff0c;然后别忘了在镜像源的“inst…

机器学习——支持向量机(SVM)

机器学习——支持向量机&#xff08;SVM&#xff09; 文章目录 前言一、SVM算法原理1.1. SVM介绍1.2. 核函数&#xff08;Kernel&#xff09;介绍1.3. 算法和核函数的选择1.4. 算法步骤1.5. 分类和回归的选择 二、代码实现&#xff08;SVM&#xff09;1. SVR&#xff08;回归&a…

【JUC系列-05】通过源码分析AQS和ReentrantLock的底层原理

JUC系列整体栏目 内容链接地址【一】深入理解JMM内存模型的底层实现原理https://zhenghuisheng.blog.csdn.net/article/details/132400429【二】深入理解CAS底层原理和基本使用https://blog.csdn.net/zhenghuishengq/article/details/132478786【三】熟练掌握Atomic原子系列基本…

android studio platform使用体验分享(as无法跳转c/c++等native源码的福音,强烈推荐)

hi&#xff0c;粉丝朋友们&#xff1a; 大家好&#xff01;这些天粉丝朋友们分享了一下Android Studio for Platform 这个最新的google开发的阅读aosp源码的工具&#xff0c;特别适合做原生系统开发。具体官方介绍如下地址&#xff1a; 参考链接&#xff1a;https://developer.…

Cookie和Session有什么区别和关系?

在技术面试中&#xff0c;经常被问到“Cookie和Session的区别”&#xff0c;大家都知道一些&#xff0c;Session比Cookie安全&#xff0c;Session是存储在服务器端的&#xff0c;Cookie是存储在客户端的&#xff0c;然而如果让你更详细地说明&#xff0c;恐怕就不怎么清楚了。 …

包装类、多线程的基本使用

包装类 1.基本数据类型对应的引用数据类型(包装类) 1.概述:所谓的包装类就是基本类型对应的类(引用类型),我们需要将基本类型转成包装类,从而让基本类型具有类的特性(说白了,就是将基本类型的数据转成包装类,就可以使用包装类中的方法来操作此数据)2.为啥要学包装类:a.将来有…

pyechart练习(一):画图小练习

1、使用Map制作全球人口分布图 import math import osimport matplotlib.pyplot as plt from pyecharts.charts import Map from pyecharts import options as opts# 只有部分国家的人口数据 POPULATION [["China", 1420062022], ["India", 1368737513],…

AI文本创作在百度App发文的实践

作者 | 内容生态端团队 导读 大语言模型&#xff08;LLM&#xff09;指包含数百亿&#xff08;或更多&#xff09;参数的语言模型&#xff0c;这些模型通常在大规模数据集上进行训练&#xff0c;以提高其性能和泛化能力。在内容创作工具接入文心一言AI能力后&#xff0c;可以为…

大数据课程L2——网站流量项目的算法分析数据处理

文章作者邮箱:yugongshiye@sina.cn 地址:广东惠州 ▲ 本章节目的 ⚪ 了解网站流量项目的算法分析; ⚪ 了解网站流量项目的数据处理; 一、项目的算法分析 1. 概述 网站流量统计是改进网站服务的重要手段之一,通过获取用户在网站的行为,可以分析出哪些内…

学习Bootstrap 5的第八天

目录 加载器 彩色加载器 实例 闪烁加载器 实例 加载器大小 实例 加载器按钮 实例 分页 分页的基本结构 实例 活动状态 实例 禁用状态 实例 分页大小 实例 分页对齐 实例 面包屑&#xff08;Breadcrumbs&#xff09; 实例 加载器 彩色加载器 在 Bootstr…