动态规划 Leetcode 322 Coin Change(零钱兑换)

题目

给定不同面额的硬币 coins 和一个总金额 amount。编写一个函数来计算可以凑成总金额所需的最少的硬币个数。如果没有任何一种硬币组合能组成总金额,返回 -1。

链接(中文版):https://leetcode-cn.com/problems/coin-change

链接(英文版):https://leetcode.com/problems/coin-change

分析

用动态规划解题就是求出递推公式,对于本题,想求总金额为amount的答案(记为Answer(amount)),需要使用总金额小于amount的答案,即递推。

对某个硬币(记面额为coin),如果小于等于amount,则说明这个硬币可以组成amount,此时有两个答案:

使用该硬币的答案Answer(amount-coin)+1,其中Answer(amount-coin)是总金额为amount-coin(小于amount)的答案,+1表示使用coin这个硬币,所以硬币数量加1。

不使用该硬币的答案Answer(amount),即当前已有答案。初始化时,我们要把这个初始答案设为一个很大的数字,这里用amlount+1就行,因为最小的面额是1,如果最后Answer(amount)==amount+1,说明给定的硬币没法组成amount。

对于这两个答案,我们取较小的,赋给Answer(amount),然后继续判断其他的硬币,当全部硬币都处理完,就得到最优的Answer(amount)

代码

class Solution:
    def coinChange(self, coins: List[int], amount: int) -> int:
        results = [amount + 1] * (amount + 1) #初始化全部答案为amount+1
        results[0] = 0 #第0个答案是0,循环从1开始
        for i in range(1, amount + 1): #依次计算1到amount的最优答案。这里是amount+1,因为range是左开右闭的
            for coin in coins: #每个最优答案的求取,都需要遍历全部的硬币
                if i >= coin: #如果某个硬币小于i,说明它可以是组成i的一部分
                    results[i] = min(results[i], results[i-coin] + 1) #是否使用这个硬币,取决于两个答案的大小
        # print(results) #至此,从1到amount的全部最优答案都有了
        return results[-1] if results[-1] <= amount else -1 #返回最有一个答案,即所求答案,如果它没有被更新过,说明它无法被组成,返回-1

 

  • 3
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值