• 每日一题 518零钱兑换2(完全背包)


    题目

    给你一个整数数组 coins 表示不同面额的硬币,另给一个整数 amount 表示总金额。

    请你计算并返回可以凑成总金额的硬币组合数。如果任何硬币组合都无法凑出总金额,返回 0 。

    假设每一种面额的硬币有无限个。

    题目数据保证结果符合 32 位带符号整数。

    示例 1:

    输入:amount = 5, coins = [1, 2, 5]
    输出:4
    解释:有四种方式可以凑成总金额:
    5=5
    5=2+2+1
    5=2+1+1+1
    5=1+1+1+1+1
    示例 2:

    输入:amount = 3, coins = [2]
    输出:0
    解释:只用面额 2 的硬币不能凑成总金额 3 。
    示例 3:

    输入:amount = 10, coins = [10]
    输出:1

    提示:

    1 <= coins.length <= 300
    1 <= coins[i] <= 5000
    coins 中的所有值 互不相同
    0 <= amount <= 5000

    题解

    记忆化搜索

    class Solution {
        private int[] coins;
        private int[][] cache;
    
        public int change(int amount, int[] coins) {
            this.coins = coins;
            int n = coins.length;
            cache = new int[n][amount + 1];
            for (int i = 0; i < n; i++) {
                Arrays.fill(cache[i],-1);
            }
            return dfs(n - 1, amount);
        }
    
        public int dfs(int i, int c) {
            if (i < 0) {
                return c == 0 ? 1 : 0;
            }
            if (cache[i][c] != -1) {
                return cache[i][c];
            }
            if (c < coins[i]) {
                return cache[i][c] = dfs(i - 1, c);
            }
            return cache[i][c] = dfs(i - 1, c) + dfs(i, c - coins[i]);
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27

    1:1递推

    class Solution {
        public int change(int amount, int[] coins) {
            int n = coins.length;
            int[][] f = new int[n + 1][amount + 1];
            f[0][0] = 1;
            for (int i = 0; i < n; i++) {
                for (int c = 0; c <= amount; c++) {
                    if (c < coins[i]) {
                        f[i + 1][c] = f[i][c];
                    } else {
                        f[i + 1][c] = f[i][c] + f[i + 1][c - coins[i]]; 
                    }
                }
            }
            return f[n][amount];
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17

    空间优化

    class Solution {
        public int change(int amount, int[] coins) {
            int n = coins.length;
            int[] f = new int[amount + 1];
            f[0] = 1;
            for (int x : coins) {
                for (int c = x; c <= amount; c++) {
                    f[c] += f[c - x];
                }
            }
            return f[amount];
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
  • 相关阅读:
    HummerRisk V0.5.2:升级对象存储、云检测、云审计和K8s资源态势等
    气膜建筑膜材分为哪些类型?
    判定转状态+序列问题上树形dp:0909T2
    骨传导耳机是利用什么原理听歌?什么骨传导耳机好用
    Android Studio 直接获取Spinner的值
    基于粒子群算法训练常规自动编码器附Matlab代码
    koa 和 express 的对比
    借口总比办法多
    什么是RTC
    spark临时文件较大问题处理
  • 原文地址:https://blog.csdn.net/fffffall/article/details/133601990