加载中...
剑指 Offer II 103-最少的硬币数目
发表于:2021-12-03 | 分类: 中等
字数统计: 915 | 阅读时长: 4分钟 | 阅读量:

原文链接: https://leetcode-cn.com/problems/gaM7Ch

中文题目

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

你可以认为每种硬币的数量是无限的。

 

示例 1:

输入:coins = [1, 2, 5], amount = 11
输出:3 
解释:11 = 5 + 5 + 1

示例 2:

输入:coins = [2], amount = 3
输出:-1

示例 3:

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

示例 4:

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

示例 5:

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

 

提示:

  • 1 <= coins.length <= 12
  • 1 <= coins[i] <= 231 - 1
  • 0 <= amount <= 104

 

注意:本题与主站 322 题相同: https://leetcode-cn.com/problems/coin-change/

通过代码

高赞题解

完全背包问题

如果把每种面额的硬币看成一种物品,而将目标总额看作是背包容量,因为硬币可以无限取,所以该问题其本质就是完全背包问题。

用函数 f(i, j) 表示用前 i 种硬币凑成的总额为 j 需要的最少数目。当使用 0 枚标号为 i - 1 的硬币时,f(i, j) 等于 f(i - 1, j) ;当使用 1 枚标号为 i - 1 的硬币时,f(i, j) 等于 f(i - 1, j - coins[i - 1]);当使用 k 枚标号为 i - 1 的硬币时,f(i, j) 等于 f(i - 1, j - k * coins[i - 1]);由于目标是求出硬币数目的最小值,因此条件转移方程为
image.png
将基本思路中求解 f(i, j - k * coins[i - 1]) 的状态转移方程显式地写出来,代入原方程中,会发现该方程可以等价地变形成这种形式。可以理解为对于同一种硬币,在当前选择的基础上再进行选择。
image.png
完整的代码如下,若硬币的种类为 n,目标额度为 t,那么时间复杂度是 O(nt),空间复杂度为 O(t)。

class Solution {
public:
    int coinChange(vector<int>& coins, int amount) {
        vector<int> dp(amount + 1, amount + 1);
        dp[0] = 0;

        for (auto& coin : coins) {
            for (int j = coin; j <= amount; ++j) {
                dp[j] = min(dp[j], dp[j - coin] + 1);
            }
        }
        return dp[amount] == amount + 1 ? -1 : dp[amount];
    }
};

另一种思路

用函数 f(i) 表示凑出总额为 i 的硬币需要的最少数目。为了凑出总额为 i 的硬币,有如下选择:在总额为 i - coins[0] 的硬币中添加 1 枚标号为 0 的硬币,此时 f(i) 等于 f(i - coins[0]) + 1;在总额为 i - coins[1] 的硬币中添加 1 枚标号为 1 的硬币,此时 f(i) 等于 f(i - coins[1]) + 1;依次类推,在总额为 i - coins[n - 1] 的硬币中添加 1 枚标号为 n - 1 的硬币,此时 f(i) 等于 f(i - coins[n - 1]) + 1;由于目标是选择最小值,所以可得转移状态方程为
image.png
完整的代码如下,若硬币的种类为 n,目标额度为 t,那么时间复杂度是 O(nt),空间复杂度为 O(t)。

class Solution {
public:
    int coinChange(vector<int>& coins, int amount) {
        vector<int> dp(amount + 1);

        for (int i = 1; i <= amount; ++i) {
            dp[i] = amount + 1;
            for (auto& coin : coins) {
                if (i >= coin) {
                    dp[i] = min(dp[i], dp[i - coin] + 1);
                }
            }
        }
        return dp[amount] == amount + 1 ? -1 : dp[amount];
    }
};

统计信息

通过次数 提交次数 AC比率
4245 8612 49.3%

提交历史

提交时间 提交结果 执行时间 内存消耗 语言
上一篇:
剑指 Offer II 102-加减的目标值
下一篇:
剑指 Offer II 104-排列的数目
本文目录
本文目录