【Golang】LeetCode 322. 零钱兑换

322. 零钱兑换

题目描述

思路

「零钱兑换」这道题目的解决思路几乎与「完全平方数」相同。主要的区别在于,「完全平方数」要求我们求表示一个数值最少需要多少个"完全平方数",相当于硬币的面额就是自然数的"完全平方数"。而「零钱兑换」这道题预先给定了我们可以使用的零钱的面额。

我们使用动态规划来解决这道题,首先初始化一个数组dp,它的长度是amount + 1,用来表示数值为i的面额最少需要多少个零钱来凑出。我们初始化i == 0之外每一个dp[i]的值为amount + 1,经过对dp数组的维护之后,如果dp[i] == amount + 1,就代表这个面额无法通过当前给定的零钱的面额表示出来(比如i == 3,但是只有面额2, 4, 6, 8可用)。基于解决「完全平方数」的经验,我们不难直接推导出维护dp数组的状态转移方程,那就是dp[i] = min(dp[i - coins[j]] + 1, dp[i])。我们使用两层循环来对dp进行维护,i遍历的是当前要表示的面额的数值,在最外层循环;j是第二层循环,用于对给定的零钱的面额值进行遍历。只有当i >= coins[j]的时候,才需要对dp进行维护。

需要注意的点是,在初始化dp[i] = amount + 1时,一定不能设置dp[0]的值也是amount + 1。对于面额0,从dp数组的表示意义上来说,它的值就应该设置为0,因为面额为0的状态不需要任何零钱来凑出。

基于以上思路和状态转移方程,我们就可以开始解题了。

Golang 题解

go 复制代码
func coinChange(coins []int, amount int) int {
    n := len(coins)
    dp := make([]int, amount + 1)
    for i := 1; i <= amount; i ++ {
        dp[i] = amount + 1
    }

    for i := 1; i <= amount; i ++ {
        for j := 0; j < n; j ++ {
            if i >= coins[j] {
                dp[i] = min(dp[i - coins[j]] + 1, dp[i])
            }
        }
    }

    if dp[amount] == amount + 1 {
        return -1
    }
    return dp[amount]
}
相关推荐
老鼠只爱大米7 小时前
LeetCode算法题详解 189:轮转数组
leetcode·轮转数组·数组旋转·环状替换法·算法面试题
葫三生8 小时前
三生原理范畴语法表明中国哲学可为算法母语
人工智能·深度学习·算法·transformer
D_FW8 小时前
数据结构第五章:树与二叉树
数据结构·算法
WHS-_-20228 小时前
Tx and Rx IQ Imbalance Compensation for JCAS in 5G NR
javascript·算法·5g
jinmo_C++8 小时前
Leetcode_59. 螺旋矩阵 II
算法·leetcode·矩阵
夏鹏今天学习了吗8 小时前
【LeetCode热题100(81/100)】零钱兑换
算法·leetcode·职场和发展
北京地铁1号线8 小时前
Embedding 模型的经典benchmark:MTEB
算法
焦糖玛奇朵婷9 小时前
盲盒小程序:开发视角下的功能与体验
java·大数据·jvm·算法·小程序
QiZhang | UESTC9 小时前
【豆包生成,写项目看】探寻最优学习路径:线性回归从框架补全到从零手写
学习·算法·线性回归
知乎的哥廷根数学学派10 小时前
基于多物理约束融合与故障特征频率建模的滚动轴承智能退化趋势分析(Pytorch)
人工智能·pytorch·python·深度学习·算法·机器学习