力扣打卡每日一题————零钱兑换

零钱兑换问题

一、问题描述

题目要求

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

示例

  • 输入:coins = [1,2,5], amount = 11

  • 输出:3(解释:11 = 5 + 5 + 1,最少需要 3 枚硬币)

  • 输入:coins = [2], amount = 3

  • 输出:-1(解释:无法用面额 2 的硬币凑出 3)

核心难点

  1. 需找到 "最少硬币数",而非所有组合数;
  2. 需处理 "无法凑出金额" 的边界情况;
  3. 避免数组越界、数值溢出等编码错误。

二、解题思路:动态规划(DP)

1. 状态定义

定义 dp[i] 表示:凑成总金额 i 所需的最少硬币数量

2. 初始化

  • 基准条件:dp[0] = 0,因为凑 0 元不需要任何硬币。
  • 其他状态:初始化 dp[i] = amount + 1amount+1 是一个 "大于最大可能值" 的数,代表 "初始不可达"。最大可能值是 amount(全用 1 元硬币),因此 amount+1 可标记为 "未更新"。

3. 状态转移

对于每个金额 i(从 1 到amount),遍历所有硬币面额 coin

  • coin <= i(当前硬币面额不超过目标金额),则:dp[i] = min(dp[i], dp[i - coin] + 1)
  • 解释:dp[i - coin] 是凑出 i - coin 元的最少硬币数,加 1 枚当前硬币即可凑出 i 元,取所有可能中的最小值。

4. 结果判断

  • dp[amount] > amount:说明始终未更新,无法凑出金额,返回 -1
  • 否则:返回 dp[amount](最少硬币数)。

三、完整代码实现

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

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

四、复杂度分析

  • 时间复杂度O(amount * len(coins))外层循环遍历 amount 个金额,内层循环遍历所有硬币,总次数为两者乘积。
  • 空间复杂度O(amount)仅需维护一个大小为 amount + 1 的 dp 数组,属于线性空间。

五、总结

问题的核心是动态规划的状态转移思想:将 "凑金额 i" 的大问题,拆解为 "凑金额 i-coin" 的小问题

相关推荐
测试19984 小时前
软件测试 - 单元测试总结
自动化测试·软件测试·python·测试工具·职场和发展·单元测试·测试用例
心中有国也有家7 小时前
cann-recipes-infer:昇腾 NPU 推理的“菜谱集合”
经验分享·笔记·学习·算法
绝知此事8 小时前
【算法突围 01】线性结构与哈希表:后端开发的收纳术
java·数据结构·算法·面试·jdk·散列表
碧海银沙音频科技研究院8 小时前
通话AEC与语音识别AEC的软硬回采链路
深度学习·算法·语音识别
csdn_aspnet8 小时前
Python 算法快闪 LeetCode 编号 70 - 爬楼梯
python·算法·leetcode·职场和发展
一只机电自动化菜鸟10 小时前
一建机电备考笔记(40) 建筑机电施工—排水管道施工(含考频+题型)
经验分享·笔记·学习·职场和发展·课程设计
m0_6294947311 小时前
LeetCode 热题 100-----26.环形链表 II
数据结构·算法·leetcode·链表
壹号用户11 小时前
用队列实现栈
数据结构·算法
做人求其滴12 小时前
面试经典 150 题 380 274
c++·算法·面试·职场和发展·力扣