0

0

递归解决有限硬币组合求和问题:优化与常见陷阱

花韻仙語

花韻仙語

发布时间:2025-09-23 11:10:01

|

149人浏览过

|

来源于php中文网

原创

递归解决有限硬币组合求和问题:优化与常见陷阱

本文探讨如何使用递归解决有限硬组合求和问题,即判断给定一组只能使用一次的硬币能否凑成特定目标金额。我们将分析原始实现中的数组复制错误和效率问题,并提出一种基于“包含或排除”策略的优化递归方案,显著提升代码的清晰度和性能,同时强调递归解法中的关键考量点。

问题描述:有限硬币组合求和

“有限硬币组合求和”问题要求我们判断,给定一组面额各异的硬币(每种硬币只能使用一次),能否凑成一个特定的目标总和。例如,给定硬币 {1, 5, 16} 和目标金额 6,我们可以用 1 + 5 凑成,因此结果为真。如果目标金额是 8,则无法凑成,结果应为假。这是一个典型的子集和问题变种,通常可以通过递归或动态规划解决。

原始递归尝试与常见陷阱

在尝试解决此类问题时,初学者常会采用一种直观的递归思路:遍历所有硬币,对于当前硬币,如果目标金额大于或等于它,就尝试将其包含在内,然后递归处理剩余硬币和减去当前硬币后的目标金额。

然而,在这种实现中,存在两个常见的陷阱:

  1. 数组复制错误: 在递归调用中,为了模拟“使用一次”的限制,需要将当前硬币从硬币列表中移除。如果手动复制数组,很容易出现索引错误。例如,原始代码中的 red[it] = coins[i] 是一个典型的错误。在构建新数组 red 时,意图是复制除 coins[i] 之外的所有元素,但正确的做法应该是复制 coins[x],其中 x 是遍历原始 coins 数组的索引。即 red[it] = coins[x] 才是正确的复制方式。
  2. 效率问题: 每次递归调用都通过循环遍历硬币数组,并在循环内部创建一个新的、长度减一的数组。这种做法不仅增加了代码的复杂性,也带来了显著的性能开销。每次递归层级都会进行不必要的数组创建和元素复制,导致时间复杂度远超预期。

考虑以下原始代码片段中的错误示例:

// 错误示例:数组复制逻辑有误
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]; // 此处应为 red[it] = coins[x];
                it += 1;
            }
        }
        ans = go(red, goal - coins[i]);
    }
}

这个错误会导致新数组 red 中填充的都是被跳过的 coins[i] 的值,而不是原始数组中其他硬币的值,从而产生不正确的结果。

优化后的递归策略:包含或排除

解决此类问题的更优雅且高效的递归方法是采用“包含或排除”策略。对于当前考虑的硬币(通常是数组的第一个元素),我们有两种选择:

  1. 不包含当前硬币: 我们跳过当前硬币,直接递归处理剩余的硬币和不变的目标金额。
  2. 包含当前硬币: 我们使用当前硬币,然后递归处理剩余的硬币和减去当前硬币面额后的目标金额。

只要这两种情况中的任何一种能够成功凑成目标金额,那么总和就是可达的。这种方法避免了复杂的循环和手动数组复制,而是通过递归参数的巧妙设计来处理子问题。

PixVerse
PixVerse

PixVerse是一款强大的AI视频生成工具,可以轻松地将多种输入转化为令人惊叹的视频。

下载

核心思想:

  • 基本情况 (Base Cases):
    • 如果目标金额 goal 为 0,说明已经成功凑成,返回 true。
    • 如果硬币列表 coins 为空或者目标金额 goal 小于 0(说明超出了目标),则无法凑成,返回 false。
  • 递归步骤 (Recursive Step):
    • 取出当前硬币 coins[0]。
    • 创建新的硬币列表 tailOfCoins,包含 coins 中除 coins[0] 之外的所有硬币。
    • 递归调用 go(tailOfCoins, goal):表示不使用 coins[0],尝试用剩余硬币凑成 goal。
    • 递归调用 go(tailOfCoins, goal - coins[0]):表示使用 coins[0],尝试用剩余硬币凑成 goal - coins[0]。
    • 如果上述任一调用返回 true,则最终结果为 true。

这种方法的优势在于:

  • 清晰简洁: 逻辑更易于理解和实现。
  • 避免手动复制错误: 利用 Arrays.copyOfRange 等工具函数可以安全高效地创建子数组。
  • 效率提升: 虽然时间复杂度仍为指数级 (O(2^N),其中 N 是硬币数量),但它避免了在每次循环迭代中重复创建数组,从而减少了常数因子,提高了实际运行效率。

核心代码实现

以下是采用“包含或排除”策略的优化递归实现:

import java.util.Arrays; // 引入 Arrays 类用于数组操作

public class FiniteCoinsSum {

    /**
     * 判断给定一组硬币(每枚硬币只能使用一次)能否凑成目标金额。
     *
     * @param coins 硬币面额数组。
     * @param goal 目标金额。
     * @return 如果能凑成目标金额,返回 true;否则返回 false。
     */
    public static boolean canMakeSum(int[] coins, int goal) {
        // 基本情况 1: 如果目标金额为0,说明已经成功凑成。
        if (goal == 0) {
            return true;
        }
        // 基本情况 2:
        // 如果硬币列表为空(没有硬币可用),或者目标金额小于0(超出了目标),
        // 则无法凑成。
        if (coins.length == 0 || goal < 0) {
            return false;
        }

        // 递归步骤:
        // 1. 获取当前硬币(数组的第一个元素)
        int currentCoin = coins[0];
        // 2. 创建一个新数组,包含除当前硬币之外的所有硬币
        int[] remainingCoins = Arrays.copyOfRange(coins, 1, coins.length);

        // 3. 两种可能性:
        //    a) 不使用当前硬币:递归调用 canMakeSum(remainingCoins, goal)
        //    b) 使用当前硬币:递归调用 canMakeSum(remainingCoins, goal - currentCoin)
        // 只要其中一种情况能凑成目标,就返回 true。
        return canMakeSum(remainingCoins, goal) || canMakeSum(remainingCoins, goal - currentCoin);
    }

    public static void main(String[] args) {
        // 测试案例
        int[] coins1 = {1, 5, 16};
        int goal1 = 6; // 1 + 5 = 6
        System.out.println("Coins: " + Arrays.toString(coins1) + ", Goal: " + goal1 + " -> " + canMakeSum(coins1, goal1)); // 预期: true

        int[] coins2 = {111, 1, 2, 3, 9, 11, 20, 30};
        int goal2 = 8; // 无法凑成 8
        System.out.println("Coins: " + Arrays.toString(coins2) + ", Goal: " + goal2 + " -> " + canMakeSum(coins2, goal2)); // 预期: false

        int[] coins3 = {2, 3, 5};
        int goal3 = 7; // 2 + 5 = 7
        System.out.println("Coins: " + Arrays.toString(coins3) + ", Goal: " + goal3 + " -> " + canMakeSum(coins3, goal3)); // 预期: true

        int[] coins4 = {10, 20, 30};
        int goal4 = 5; // 无法凑成
        System.out.println("Coins: " + Arrays.toString(coins4) + ", Goal: " + goal4 + " -> " + canMakeSum(coins4, goal4)); // 预期: false

        int[] coins5 = {1, 2, 3};
        int goal5 = 0; // 目标为0,直接返回true
        System.out.println("Coins: " + Arrays.toString(coins5) + ", Goal: " + goal5 + " -> " + canMakeSum(coins5, goal5)); // 预期: true
    }
}

注意事项与总结

  • 递归基的准确性: 正确定义递归的终止条件至关重要。goal == 0 是成功条件,而 coins.length == 0 || goal < 0 是失败条件。
  • 数组的不可变性与子数组创建: 在递归中传递数组时,通常需要确保每次递归调用都处理一个“新”的子问题状态。使用 Arrays.copyOfRange 可以方便地创建子数组,避免原始数组被修改,这对于递归的正确性至关重要。
  • 时间复杂度: 尽管优化后的代码更简洁,但其时间复杂度仍为指数级 (O(2^N)),对于大规模的硬币数量 N,性能可能成为瓶颈。在这种情况下,可以考虑使用动态规划(背包问题变种)来优化到伪多项式时间复杂度。
  • 问题建模: 许多组合问题都可以抽象为“包含或排除”某个元素,然后递归解决子问题。熟练掌握这种思维模式有助于解决多种类似问题。

通过采用这种优化的递归策略,我们不仅修复了原始代码中的数组复制错误,还显著提升了代码的清晰度和可维护性,为解决有限硬币组合求和问题提供了一个高效且易于理解的递归解决方案。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
length函数用法
length函数用法

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

954

2023.09.19

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

25

2026.03.13

Python异步编程与Asyncio高并发应用实践
Python异步编程与Asyncio高并发应用实践

本专题围绕 Python 异步编程模型展开,深入讲解 Asyncio 框架的核心原理与应用实践。内容包括事件循环机制、协程任务调度、异步 IO 处理以及并发任务管理策略。通过构建高并发网络请求与异步数据处理案例,帮助开发者掌握 Python 在高并发场景中的高效开发方法,并提升系统资源利用率与整体运行性能。

44

2026.03.12

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

177

2026.03.11

Go高并发任务调度与Goroutine池化实践
Go高并发任务调度与Goroutine池化实践

本专题围绕 Go 语言在高并发任务处理场景中的实践展开,系统讲解 Goroutine 调度模型、Channel 通信机制以及并发控制策略。内容包括任务队列设计、Goroutine 池化管理、资源限制控制以及并发任务的性能优化方法。通过实际案例演示,帮助开发者构建稳定高效的 Go 并发任务处理系统,提高系统在高负载环境下的处理能力与稳定性。

50

2026.03.10

Kotlin Android模块化架构与组件化开发实践
Kotlin Android模块化架构与组件化开发实践

本专题围绕 Kotlin 在 Android 应用开发中的架构实践展开,重点讲解模块化设计与组件化开发的实现思路。内容包括项目模块拆分策略、公共组件封装、依赖管理优化、路由通信机制以及大型项目的工程化管理方法。通过真实项目案例分析,帮助开发者构建结构清晰、易扩展且维护成本低的 Android 应用架构体系,提升团队协作效率与项目迭代速度。

92

2026.03.09

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

102

2026.03.06

Rust内存安全机制与所有权模型深度实践
Rust内存安全机制与所有权模型深度实践

本专题围绕 Rust 语言核心特性展开,深入讲解所有权机制、借用规则、生命周期管理以及智能指针等关键概念。通过系统级开发案例,分析内存安全保障原理与零成本抽象优势,并结合并发场景讲解 Send 与 Sync 特性实现机制。帮助开发者真正理解 Rust 的设计哲学,掌握在高性能与安全性并重场景中的工程实践能力。

227

2026.03.05

PHP高性能API设计与Laravel服务架构实践
PHP高性能API设计与Laravel服务架构实践

本专题围绕 PHP 在现代 Web 后端开发中的高性能实践展开,重点讲解基于 Laravel 框架构建可扩展 API 服务的核心方法。内容涵盖路由与中间件机制、服务容器与依赖注入、接口版本管理、缓存策略设计以及队列异步处理方案。同时结合高并发场景,深入分析性能瓶颈定位与优化思路,帮助开发者构建稳定、高效、易维护的 PHP 后端服务体系。

530

2026.03.04

热门下载

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

精品课程

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

共23课时 | 4.4万人学习

C# 教程
C# 教程

共94课时 | 11.3万人学习

Java 教程
Java 教程

共578课时 | 82.1万人学习

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

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