给你一个整数数组 coins ,表示不同面额的硬币;以及一个整数 amount ,表示总金额。
计算并返回可以凑成总金额所需的 最少的硬币个数 。如果没有任何一种硬币组合能组成总金额,返回 -1 。
你可以认为每种硬币的数量是无限的
示例 1:
输入:coins = [1, 2, 5], amount = 11
输出:3
解释:11 = 5 + 5 + 1
package MidTest;
public class 零钱兑换 {
public static void main(String[] args) {
int[] coins = {186,419,83,408};
int amount = 6249;
System.out.println(coinChange(coins,amount));
}
static int[] memo;
public static int coinChange(int[] coins, int amount) {
if(coins.length==0) return -1;
memo = new int[amount];
return findway(coins,amount);
}
private static int findway(int[] coins, int amount) {
if(amount<0) return -1;
if(amount==0) return 0;
//如果当前amount的数在memo数组中出现过,直接调用。为什么下标是amount-1,因为数组从0开始啊,笨逼
if(memo[amount-1]!=0) return memo[amount-1];
int min = Integer.MAX_VALUE;
for (int i = 0; i < coins.length; i++) {
int res = findway(coins,amount-coins[i]);
if(res>=0&&res<min){
min = res+1;
}
}
memo[amount-1] = min==Integer.MAX_VALUE?-1:min;
return memo[amount-1];
}
}
findway函数返回的是当前amount所需的硬币数(其实就是返回的memo数组中的值),-1则是无法用该硬币数组组成amount。
memo数组是用于递归中的记忆数组,memo[amount-1]表示amount所需的硬币数,当在递归中重复出现相同的amount时,就可以直接调用。
min=res+1是加上最后一次findway返回0时(即amount=0)兑换的那枚硬币。