0

0

优化递归算法:解决有限硬币凑数问题

花韻仙語

花韻仙語

发布时间:2025-09-23 10:51:18

|

720人浏览过

|

来源于php中文网

原创

优化递归算法:解决有限硬币凑数问题

本文探讨了如何使用递归算法解决给定有限数量硬(每种面额一枚)能否凑成目标金额的问题。文章首先分析了常见递归实现中的潜在错误,特别是数组复制逻辑,随后提出并详细解释了一种更简洁、高效的递归策略。该策略通过处理当前硬币的两种可能性(使用或不使用)来避免复杂循环,并提供了优化后的代码示例。

问题描述:有限硬币凑数

假设我们有一组不同面额的硬币,每种面额只有一枚。我们的任务是判断是否能够从这组硬币中选出若干枚,使其总和恰好等于一个给定的目标金额。例如,给定硬币面额 [1, 5, 16] 和目标金额 6,我们可以选择 1 和 5 凑成 6,因此结果为真。

初始递归尝试及问题分析

一种直观的递归思路是,遍历所有可用的硬币。对于每一枚硬币,我们尝试将其从集合中移除,并递归地判断剩余硬币能否凑成 目标金额 - 当前硬币面额。如果目标金额恰好为零,则说明找到了一个组合;如果目标金额变为负数或硬币用尽但目标金额不为零,则说明此路径不可行。

以下是初始代码实现的一个示例:

boolean go(int[] coins, int goal) {
  boolean ans = false;
  // 基本情况:目标金额为0,成功
  if (goal == 0) {
    return true;
  } else if (goal < 0) { // 基本情况:目标金额为负数,失败
    return false;
  }

  // 遍历所有硬币
  for (int i = 0; i < coins.length && (!ans); i++) {
    if (goal >= coins[i]) {
      // 创建一个新数组,移除当前硬币
      int[] red = new int[coins.length - 1];
      int it = 0;
      for (int x = 0; x < coins.length; x++) {
        if (!(i == x)) {
          // 错误所在:这里应该复制 coins[x],而不是 coins[i]
          red[it] = coins[i]; // 这一行是错误的根源
          it += 1;
        }
      }
      // 递归调用,目标金额减去当前硬币面额
      ans = go(red, goal - coins[i]);
    }
  }
  return ans;
}

这段代码在某些测试用例中会产生错误。例如,对于硬币 [111, 1, 2, 3, 9, 11, 20, 30] 和目标金额 8,它错误地返回 true,而正确结果应为 false。

关键错误定位与解析

问题的核心在于数组 red 的复制逻辑。在内部循环中,当试图将 coins 数组中除 coins[i] 以外的元素复制到 red 数组时,代码错误地写成了 red[it] = coins[i]。这意味着无论 x 是什么,它都会将 coins[i](即当前被“移除”的硬币)的值复制到 red 数组中,而不是将 coins[x](即其他未被移除的硬币)的值复制进去。

正确的复制方式应该是 red[it] = coins[x],以确保 red 数组包含的是除了 coins[i] 之外的所有硬币。这个错误导致 red 数组的内容不正确,从而使后续的递归调用基于错误的数据进行计算。

优化递归策略:选择与不选择

为了避免复杂的数组复制操作并提高代码的清晰度,我们可以采用一种更优雅的递归策略。对于当前考虑的硬币 coins[0](假设我们总是处理数组的第一个元素),我们只有两种选择:

Tanka
Tanka

具备AI长期记忆的下一代团队协作沟通工具

下载
  1. 不使用当前硬币 coins[0]: 这种情况下,我们递归地在剩余的硬币 coins[1...n-1] 中寻找目标金额 goal。
  2. 使用当前硬币 coins[0]: 这种情况下,我们递归地在剩余的硬币 coins[1...n-1] 中寻找目标金额 goal - coins[0]。

只要这两种情况中的任何一种能够成功凑出目标金额,那么总的结果就是 true。

优化代码实现

基于“选择与不选择”的策略,我们可以得到以下更简洁、高效的递归实现:

import java.util.Arrays; // 需要导入 Arrays 类

boolean go(int[] coins, int goal) {
    // 基本情况 1:目标金额为0,说明已成功凑齐
    if (goal == 0) {
        return true;
    }
    // 基本情况 2:
    // a) 硬币已用尽,但目标金额不为0,说明无法凑齐
    // b) 目标金额为负数,说明当前路径选择的硬币面额过大,无法凑齐
    else if (coins.length == 0 || goal < 0) {
        return false;
    }
    // 递归步骤
    else {
        // 创建一个新数组,包含除第一个硬币外的所有硬币
        int[] tailOfCoins = Arrays.copyOfRange(coins, 1, coins.length);

        // 两种可能性:
        // 1. 不使用当前硬币 coins[0]:在剩余硬币中寻找目标金额 goal
        // 2. 使用当前硬币 coins[0]:在剩余硬币中寻找目标金额 goal - coins[0]
        return go(tailOfCoins, goal) || go(tailOfCoins, goal - coins[0]);
    }
}

代码解析

  • 基本情况 (Base Cases):
    • if (goal == 0): 这是成功的终止条件。如果目标金额已经降到 0,说明我们找到了一种组合方式。
    • else if (coins.length == 0 || goal
    • coins.length == 0: 如果所有硬币都已考虑完毕,但 goal 仍然不为 0,则无法凑齐。
    • goal
  • 递归步骤 (Recursive Step):
    • int[] tailOfCoins = Arrays.copyOfRange(coins, 1, coins.length);: 使用 Arrays.copyOfRange 方法创建一个新数组 tailOfCoins,它包含了 coins 数组中除了第一个元素 coins[0] 之外的所有元素。这是为了在递归调用中处理“剩余硬币”。
    • return go(tailOfCoins, goal) || go(tailOfCoins, goal - coins[0]);: 这是核心的递归逻辑。
      • go(tailOfCoins, goal): 代表“不使用当前硬币 coins[0]”的情况。我们继续在 tailOfCoins 中寻找 goal。
      • go(tailOfCoins, goal - coins[0]): 代表“使用当前硬币 coins[0]”的情况。我们从 goal 中减去 coins[0] 的面额,然后在 tailOfCoins 中寻找新的目标金额。
      • || (逻辑或): 只要这两种情况中的任何一种能够返回 true(即找到一个组合),那么整个函数就返回 true。
  • 时间复杂度考量

    这种优化后的递归方法虽然仍然是指数级的(因为它会探索所有可能的组合),但它避免了在每次递归调用中循环遍历数组并重新构建新数组的开销。Arrays.copyOfRange 操作本身会产生一个新的数组副本,其时间复杂度为 O(N),其中 N 是当前数组的长度。然而,由于我们每次只处理一个硬币并将其从考虑范围中移除,相比于在循环中多次创建新数组,这种方式通常更为高效且代码更简洁。对于实际应用中可能出现的性能瓶颈,可以考虑使用动态规划(背包问题变种)来进一步优化,将其时间复杂度降低到多项式级别。

    注意事项与总结

    • 数组不可变性: 在递归中传递数组时,为了避免副作用,通常会创建数组的副本。Arrays.copyOfRange 是一个方便且安全的方法。
    • 基本情况的重要性: 正确定义递归的基本情况是避免无限递归和确保正确结果的关键。
    • 选择与不选择模式: 许多组合问题都可以通过这种“选择与不选择”的递归模式来解决,它能清晰地分解问题,并避免复杂的循环逻辑。
    • 效率考量: 尽管递归解决方案简洁优雅,但对于大规模输入,其指数级的时间复杂度可能成为瓶颈。在这种情况下,应考虑使用动态规划等优化技术。

    通过理解并应用这种优化后的递归策略,我们可以更准确、更高效地解决有限硬币凑数问题,同时避免常见的编程陷阱。

    相关专题

    更多
    if什么意思
    if什么意思

    if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

    757

    2023.08.22

    string转int
    string转int

    在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

    338

    2023.08.02

    int占多少字节
    int占多少字节

    int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

    542

    2024.08.29

    c++怎么把double转成int
    c++怎么把double转成int

    本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

    53

    2025.08.29

    C++中int的含义
    C++中int的含义

    本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

    197

    2025.08.29

    length函数用法
    length函数用法

    length函数用于返回指定字符串的字符数或字节数。可以用于计算字符串的长度,以便在查询和处理字符串数据时进行操作和判断。 需要注意的是length函数计算的是字符串的字符数,而不是字节数。对于多字节字符集,一个字符可能由多个字节组成。因此,length函数在计算字符串长度时会将多字节字符作为一个字符来计算。更多关于length函数的用法,大家可以阅读本专题下面的文章。

    922

    2023.09.19

    页面置换算法
    页面置换算法

    页面置换算法是操作系统中用来决定在内存中哪些页面应该被换出以便为新的页面提供空间的算法。本专题为大家提供页面置换算法的相关文章,大家可以免费体验。

    403

    2023.08.14

    Java编译相关教程合集
    Java编译相关教程合集

    本专题整合了Java编译相关教程,阅读专题下面的文章了解更多详细内容。

    9

    2026.01.21

    C++多线程相关合集
    C++多线程相关合集

    本专题整合了C++多线程相关教程,阅读专题下面的的文章了解更多详细内容。

    3

    2026.01.21

    热门下载

    更多
    网站特效
    /
    网站源码
    /
    网站素材
    /
    前端模板

    精品课程

    更多
    相关推荐
    /
    热门推荐
    /
    最新课程
    Kotlin 教程
    Kotlin 教程

    共23课时 | 2.7万人学习

    C# 教程
    C# 教程

    共94课时 | 7.2万人学习

    Java 教程
    Java 教程

    共578课时 | 48.8万人学习

    关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
    php中文网:公益在线php培训,帮助PHP学习者快速成长!
    关注服务号 技术交流群
    PHP中文网订阅号
    每天精选资源文章推送

    Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号