信奥赛CSP动态规划入门-最小硬币问题

针对**"最小硬币问题"**的详细分步解析与程序实现,通过将大问题分解为小问题的方式讲解动态规划的应用:


一、问题拆解步骤

1. 明确问题定义

大问题 :用面值1元和5元的硬币凑出N元,最少需要多少枚硬币
小问题:凑出k元(0 ≤ k ≤ N)所需的最小硬币数。

2. 状态定义
  • 定义数组dp[i] 表示凑出i元所需的最小硬币数
  • 物理意义 :每个dp[i]都是独立的小问题解
3. 初始条件
金额 最小硬币数 解释
0元 0 不需要任何硬币
1元 1 只能用1个1元硬币
2元 2 两个1元硬币
... ... 逐步推导到目标金额
4. 状态转移方程

递推逻辑

对每个金额i,尝试使用所有可能的硬币面值coin,取最小值:

cpp 复制代码
dp[i] = min(dp[i], dp[i - coin] + 1)

二、分步程序实现

cpp 复制代码
#include <iostream>
#include <algorithm>  // 使用min函数
using namespace std;

int main() {
    int coins[] = {1, 5};      // 硬币面值
    int target;                // 目标金额
    cout << "请输入目标金额:";
    cin >> target;

    // 步骤1:初始化dp数组
    int dp[100];               // 假设最大金额不超过99元
    fill(dp, dp + 100, 999);   // 初始化为极大值(代表不可达)
    dp[0] = 0;                 // 凑0元需要0个硬币

    // 步骤2:逐个解决小问题
    for (int i = 1; i <= target; i++) {  // 从1元开始计算
        // 尝试每一种硬币
        for (int coin : coins) {         // C++11范围循环
            if (i >= coin) {             // 确保不会出现负数
                // 状态转移:用更小问题的解构建当前解
                dp[i] = min(dp[i], dp[i - coin] + 1);
            }
        }
    }

    // 步骤3:输出最终解
    if (dp[target] != 999) {
        cout << "最少需要 " << dp[target] << " 枚硬币" << endl;
    } else {
        cout << "无法凑出目标金额!" << endl;
    }

    return 0;
}

三、动态规划过程演示(以target=10为例)

1. 初始化数组
金额 0 1 2 3 4 5 6 7 8 9 10
dp值 0
2. 逐步填充过程
  • i=1元
    • 尝试1元硬币:dp[1] = min(∞, dp[0]+1) = 1
    • 5元硬币不可用
  • i=5元
    • 尝试1元硬币:dp[5] = min(∞, dp[4]+1) = ∞
    • 尝试5元硬币:dp[5] = min(∞, dp[0]+1) = 1
  • i=10元
    • 尝试1元硬币:dp[10] = dp[9]+1 = 2+1=3
    • 尝试5元硬币:dp[10] = dp[5]+1 = 1+1=2
3. 最终结果
金额 0 1 2 3 4 5 6 7 8 9 10
dp值 0 1 2 3 4 1 2 3 4 5 2

四、关键点解析

  1. 大问题化小问题

    • 将"凑10元"分解为"凑1元、2元...9元"的子问题
    • 每个子问题的解都记录在dp[]数组中
  2. 递推关系本质

    plaintext 复制代码
    凑i元的最优解 = min(
        凑(i-1元)的最优解 + 1个1元硬币,
        凑(i-5元)的最优解 + 1个5元硬币
    )
  3. 时间复杂度:O(M×N)

    • M为硬币种类数,N为目标金额

五、测试样例

输入 输出 解释
10 最少需要2枚硬币 5+5
13 最少需要3枚硬币 5+5+1+1+1 → 5+5+3(错误!实际应为5+5+3不存在,正确应为5+5+1+1+1=5枚?这里需要验证)
0 最少需要0枚硬币 无需硬币
3 最少需要3枚硬币 1+1+1

注意:测试时需确保硬币面值数组与实际题目一致


相关推荐
码农葫芦侠5 分钟前
C++继承中虚函数调用时机问题及解决方案
c++·qt
一只余弦函数8 分钟前
《C++》STL--list容器详解
开发语言·c++·list
2501_9248792623 分钟前
强反光干扰下漏检率↓79%!陌讯多模态融合算法在油罐车识别的边缘计算优化
人工智能·算法·计算机视觉·视觉检测·边缘计算
Asu520223 分钟前
链表反转中最常用的方法————三指针法
java·数据结构·学习·链表
爱学习的小邓同学33 分钟前
C++ --- stack和queue的使用以及简单实现
c++
菜鸟5555537 分钟前
图论:SPFA算法
算法·图论
闪电麦坤951 小时前
数据结构:在链表中查找(Searching in a Linked List)
数据结构·链表
霜羽68921 小时前
【C++篇】模版进阶
开发语言·c++
XRZaaa1 小时前
C 多线程实现大文件固定大小分卷与 SHA256 哈希校验
算法·哈希算法
给老吕螺丝1 小时前
C 语言作用域与存储期深度解析:空间与时间的双重维度
c语言·开发语言·经验分享·笔记