LeetCode518. 零钱兑换 II

news2025/1/11 5:12:48

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 。

示例 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 时的组合方式数量。

先遍历物品后遍历背包vs先遍历背包后遍历物品

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

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

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

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

话比较抽象,但是我把测试代码写在下面了,大家可以复制过去体会体会。

测试代码1(先遍历物品后遍历背包):

#include <iostream>
using namespace std;
#include <vector>

void print(vector<int>& vec) {
    for (auto i : vec) {
        cout << i<<" ";
    }
    cout << endl;
}
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]];
            }
            print(dp);	//i每遍历一次,打印一次dp数组
        }
        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;
}

在这里插入图片描述

测试代码2(先遍历背包后遍历物品):

#include <iostream>
using namespace std;
#include <vector>

void print(vector<int>& vec) {
    for (auto i : vec) {
        cout << i << " ";
    }
    cout << endl;
}
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]];
            }
            print(dp);
        }
        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;
}

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

RTMP流媒体服务器EasyDSS视频直播点播平台如何生成可自动播放的分享链接

EasyDSS支持一站式的上传、转码、直播、回放、嵌入、分享功能&#xff0c;具有多屏播放、自由组合、接口丰富等特点。平台可以为用户提供专业、稳定的直播推流、转码、分发和播放服务&#xff0c;全面满足超低延迟、超高画质、超大并发访问量的要求。在推流方面&#xff0c;Eas…

PostgreSQL PG15 新功能 PG_WALINSPECT

开头还是介绍一下群&#xff0c;如果感兴趣PolarDB ,MongoDB ,MySQL ,PostgreSQL ,Redis &#xff0c;Oracle ,Oceanbase 等有问题&#xff0c;有需求都可以加群群内有各大数据库行业大咖&#xff0c;CTO&#xff0c;可以解决你的问题。加群请加微信号 liuaustin3 &#xff08;…

ABAP内表排序

SORT在用于给内表排序时&#xff0c;后面可以用ASCENDING和DESCENDING进行升序和降序排列&#xff0c;但是这其中用法很多&#xff0c;经过尝试后总结如下&#xff1a; 1.SORT LT_TAB BY WERKS LGORT EMAIL. 正常排序并使用默认ASCENDING. 2.SORT LT_TAB BY WERKS LGORT EMAI…

Java测试(10)--- selenium

1.定位一组元素 &#xff08;1&#xff09;如何打开本地的HTML页面 拼成一个URL &#xff1a;file: /// 文件的绝对路径 import os os.path.abspath(文件的绝对路径&#xff09; &#xff08;2&#xff09;先定位出同一类元素&#xff08;tag name&#xff0c;name&…

NUC980webServer开发

目录 1.RTL8189FTV驱动移植 2.wifi配置工具hostapd移植 1.openssl-1.0.2r交叉编译 2.libnl-3.2.25.tar.gz交叉编译 3.hostapd-2.9.tar.gz交叉编译 4.移植相关工具到开发板 1.RTL8189FTV驱动移植 1. 把驱动文件源码放在linux源码的drivers/net/wireless/realtek/rtlwifi/目录…

LeetCode //C - 114. Flatten Binary Tree to Linked List

114. Flatten Binary Tree to Linked List Given the root of a binary tree, flatten the tree into a “linked list”: The “linked list” should use the same TreeNode class where the right child pointer points to the next node in the list and the left child …

在Android和iOS上设置手机ip详细教程

大家好&#xff01;今天我们将分享一个关于如何在Android和iOS设备上设置手机ip&#xff08;Layer 2 Tunneling Protocol&#xff09;的简易教程。如果你想要通过安全且可靠的方式连接到远程网络&#xff0c;那么跟着本文一起学习吧&#xff01;无需复杂操作&#xff0c;让我们…

KUKA机器人后台控制程序(SPS)介绍

KUKA机器人后台控制程序(SPS)介绍 KUKA机器人后台控制程序主要包括以下几部分: RC:运动控制、机器人轨迹规划 优先级1 I/O刷新:输入输出信号的控制 优先级1 SPS:用户可编辑的后台逻辑程序 优先级2 显示界面刷新:示教器显示画面的控制 优先级3 以上的程序需要12ms(固定…

企业架构LNMP学习笔记19

Nginx 第三方模块的使用&#xff1a; Nginx官方没有的功能&#xff0c;开源开发人员定制开发了一些功能&#xff0c;把代码公布出来&#xff0c;可以通过编译加载第三方模块的方式&#xff0c;使用新功能。 NGINX 3rd Party Modules | NGINX shell > tar xvf ngx-fancyinde…

CVE-2017-12149

春秋云镜 CVE-2017-12149 JBoss反序列化漏洞 靶标介绍 2017年8月30日&#xff0c;厂商Redhat发布了一个JBOSSAS 5.x 的反序列化远程代码执行漏洞通告。该漏洞位于JBoss的HttpInvoker组件中的 ReadOnlyAccessFilter 过滤器中&#xff0c;其doFilter方法在没有进行任何安全检查…

Win10怎么设置待机时间

我们在使用电脑的过程中&#xff0c;经常因为有事需要离开电脑&#xff0c;长时间不操作电脑就会进行待机睡眠状态&#xff0c;那么Win10怎么设置待机时间呢&#xff0c;下面小编就给大家详细介绍一下Win10设置待机时间的方法&#xff0c;大家感兴趣的话可以来看一看。 设置方…

新华社《中国扫描十年发展图鉴》:扫描全能王为3亿用户带去“掌心里的便利”

扫描设备从“两手搬”到“进口袋”的过程中经历了什么&#xff1f; 近日&#xff0c;新华社发布了《中国扫描十年发展图鉴》&#xff08;简称《图鉴》&#xff09;&#xff0c;对扫描设备、技术、应用领域的发展历史进行了深入盘点。《图鉴》显示&#xff0c;扫描一度是价格接…

MySQL——数据的删除以及MySQL中的约束

删除数据 删除表中的一行数据&#xff0c;也必须加上 WHERE条件&#xff0c;否则整列的数据都会被删除。删除语句&#xff1a; delete from 表名 where 条件; 他会将所有的符合条件的数据删除&#xff0c;如果不写条件&#xff0c;则表中的数据全部删除&#xff1a; 如果不添…

软件测试/测试开发丨学会与 AI 对话,高效提升学习效率

点此获取更多相关资料 简介 ChatGPT 的主要优点之一是它能够理解和响应自然语言输入。在日常生活中&#xff0c;沟通本来就是很重要的一门课程&#xff0c;沟通的过程中表达越清晰&#xff0c;给到的信息越多&#xff0c;那么沟通就越顺畅。 和 ChatGPT 沟通也是同样的道理&…

方向介绍:基于深度学习的轨迹预测

方向介绍&#xff1a;基于深度学习的轨迹预测 文章目录 方向介绍&#xff1a;基于深度学习的轨迹预测问题定义典型方法挑战未来展望参考 基于深度学习的轨迹预测是一种利用神经网络模型来预测移动物体的未来位置和运动状态的技术。这种技术在许多领域都有重要的应用&#xff0c…

Jetpack Compose 入门教程之Text

这个文本显示组件应该是我们最常用的组件,下面会非常细 归纳 实例 下面一一演示这些属性与控制逻辑 文本的展示 Text组件 所有构造方法都是text:String,要想用string.xml里面的字符串资源 得使用 stringResource方法,其相似方法如下/** Copyright 2019 The Android Open Sou…

如何挑选低值易耗品管理系统?优化企业管理效率与成本控制

在现代企业管理中&#xff0c;低值易耗品的管理是一个容易被忽视但却十分重要的环节。低值易耗品包括办公用品、耗材、工具等&#xff0c;它们虽然单价不高&#xff0c;但数量庞大且频繁使用&#xff0c;对企业的日常运营和成本控制有着重要影响。为了提高管理效率、降低成本&a…

Linux与shell命令行学习

文章目录 走进shell基本的bash shell命令2.1 遍历目录 cd2.2 查看文件和目录列表 ls2.3 创建文件 touch2.4 复制文件 cp2.5 自动补全 tab2.6 链接文件 ln2.7 文件重命名 mv2.8 删除文件 rm2.9 创建目录 mkdir2.10 删除目录 rmdir2.11 查看文件类型 file2.12 查看整个文件 cat、…

flume1.11.0安装部署

1、准备安装包apache-flume-1.11.0-bin.tar.gz&#xff1b; 上传&#xff1b; 2、安装flume-1.11.0&#xff1b; 解压&#xff1b; tar -zxvf apache-flume-1.11.0-bin.tar.gz -C /opt/server 进入conf目录&#xff0c;修改flume-env.sh&#xff0c;配置JAVA_HOME&#xff1b…

docker 生成镜像的几个问题

docker 生成镜像的几个问题 根据jdk8.tar.gz 打包Jdk8 镜像失败运行镜像报错差不多是网络ip错误,在网上说重启docker即可解决运行mysql5.7.25 镜像失败向daemon.json文件添加内容导致docker重启失败docker run 命令常用参数根据jdk8.tar.gz 打包Jdk8 镜像失败 首选做准备工作…