动态规划12-零钱兑换(Java)

文章介绍了如何使用动态规划方法解决‘零钱兑换’问题,通过创建二维dp数组来计算凑出给定总金额所需的最少硬币数量。给出了Java代码实现,包括两种版本,适用于一维和二维数组情况。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

12.零钱兑换

  • 题目描述

给你一个整数数组 coins ,表示不同面额的硬币;以及一个整数 amount ,表示总金额。计算并返回可以凑成总金额所需的 最少的硬币个数 。如果没有任何一种硬币组合能组成总金额,返回 -1

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

示例 1:

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

示例 2:

输入:coins = [2], amount = 3
输出:-1
  • 题目分析

因为每个背包可以放同一种物品多次,所以这是一道完全背包问题

动态规划五部曲

1.创立二维dp数组:dp[i][j]:总金额为j时,放入面额为coins[0-i]时最小硬币个数
2.初始化二维数组:
当i=0时,表示只可以放入coins[0]:当背包容量为j>=coins[0]时,表示可以放下coin[0],则有dp[0][j] = dp[0][j - coins[0]] + 1;//比背包容量为j-coins[i]多一枚硬币
当j<coins[0]表示放不下此硬币,则不存在可装满背包容量为j的硬币数,此时需要设立一个标识,用来处理-1的情况,这里设置为dp[0][j] = amount + 1,即dp[i][j]>amount时,返回-1
3.递推公式
装下coins[i]:dp[i][j] = Math.min(dp[i - 1][j], dp[i][j - coins[i]] + 1);//注意是最小
装不下coins[i]:dp[i][j] = dp[i - 1][j];
4.遍历顺序:先物品,再背包
5.遍历dp数组:
public static void printArray(int[][] array) {
        for (int i = 0; i < array.length; i++) {
            for (int j = 0; j < array[0].length; j++) {
                System.out.print(array[i][j] + " ");
            }
            System.out.println();
        }
    }    
  • Java代码实现(二维数组)
public static int coinChange(int[] coins, int amount) {
        int[][] dp = new int[coins.length][amount + 1];
        for (int j = 1; j <= amount; j++) {
            if (j >= coins[0]) {
                dp[0][j] = dp[0][j - coins[0]] + 1;
            } else {
                dp[0][j] = amount + 1;
            }
        }

        for (int i = 1; i < coins.length; i++) {
            for (int j = 1; j <= amount; j++) {
                if (j < coins[i]) {
                    dp[i][j] = dp[i - 1][j];
                } else {
                    dp[i][j] = Math.min(dp[i - 1][j], dp[i][j - coins[i]] + 1);
                }
            }
        }
        return dp[coins.length - 1][amount] > amount ? -1 : dp[coins.length - 1][amount];
    }
  • 降阶处理
  • 由于是一位和二维性质一样,可以将二维数组[i]的部分去掉即可
public static int coinChange(int[] coins, int amount) {
        int[] dp = new int[amount + 1];
        for (int j = 1; j <= amount; j++) {
            if (j >= coins[0]) {
                dp[j] = dp[j - coins[0]] + 1;
            } else {
                dp[j] = amount + 1;
            }
        }

        for (int i = 1; i < coins.length; i++) {
            for (int j = 1; j <= amount; j++) {
                if (j >= coins[i]) {
                    dp[j] = Math.min(dp[j], dp[j - coins[i]] + 1);
                }
            }
        }
        return dp[amount] > amount ? -1 : dp[amount];
    }
### 动态规划解决零钱兑换问题 零钱兑换问题是经典的动态规划问题之一,目标是找到最少数量的硬币来组成给定金额。如果无法完成,则返回 `-1`。 #### 方法一:基于引用[1] 以下是一个简单的 Java 实现方法: ```java import java.util.Scanner; public class CoinChange { public static void main(String[] args) { Scanner in = new Scanner(System.in); int N = in.nextInt(); // 录入零钱总额 in.close(); int[] faces = {1, 5, 20, 25}; // 可选硬币面额 int[] dp = new int[N + 1]; for (int i = 1; i <= N; i++) { int min = Integer.MAX_VALUE; for (int face : faces) { if (face > i) continue; // 如果当前硬币面值大于所需金额,则跳过 min = Math.min(dp[i - face], min); // 更新最小值 } dp[i] = min + 1; // 当前状态等于之前的状态加上一枚硬币 } System.out.println(dp[N]); // 输出最终结果 } } ``` 上述代码的核心在于 `dp` 数组的设计[^1]。数组中的每一个位置表示达到该金额所需的最少硬币数。 --- #### 方法二:基于引用[2] 另一种更通用的方法可以处理任意硬币集合和金额输入: ```java class Solution { public int coinChange(int[] coins, int amount) { int[] dp = new int[amount + 1]; Arrays.fill(dp, amount + 1); // 初始化为较大值 dp[0] = 0; // 达到金额为0时不需要任何硬币 for (int i = 1; i <= amount; i++) { for (int coin : coins) { if (i - coin >= 0) { dp[i] = Math.min(dp[i], dp[i - coin] + 1); // 加上当前硬币的数量 } } } return dp[amount] > amount ? -1 : dp[amount]; // 判断是否能够达成目标金额 } } ``` 这种方法同样利用了动态规划的思想,并且更加灵活,适用于不同的硬币种类和金额范围[^2]。 --- #### 方法三:基于引用[3] 下面是一种稍有不同的实现方式,它也采用了动态规划策略: ```java public static int coinChange(int[] coins, int amount) { if (amount < 1) { return 0; } int[] dp = new int[amount + 1]; int sum = 0; while (++sum <= amount) { int min = -1; for (int coin : coins) { if (sum >= coin && dp[sum - coin] != -1) { int temp = dp[sum - coin] + 1; min = min < 0 ? temp : (temp < min ? temp : min); } } dp[sum] = min; } return dp[amount]; } ``` 这段代码的特点是在更新过程中引入了一个临时变量 `min` 来记录最优解[^3]。 --- #### 关键点解析 - **初始化**:通常将 `dp` 数组初始化为一个较大的值(如 `amount + 1`),以便后续计算时能正确判断不可达状态。 - **转移方程**:对于每种硬币,尝试将其加入已有的组合中,从而逐步构建出到达目标金额的最佳路径。 - **边界条件**:当金额为 `0` 时,显然需要 `0` 枚硬币;而当某些子问题无解时,应标记为不可达状态(常用 `-1` 表示)[^4]。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值