• 动态规划之硬币找零问题


    问题描述:

    假设有几种硬币,并且数量无限。请找出能够组成某个数目的找零所使用最少的硬币数。例如几种硬币为[1, 3, 5], 面值2的最少硬币数为2(1, 1), 面值4的最少硬币数为2(1, 3), 面值11的最少硬币数为3(5, 5, 1或者5, 3, 3).

    问题分析:

    假设不同的几组硬币为数组coin[0, ..., n-1]. 则求面值k的最少硬币数count(k), 那么count函数和硬币数组coin满足这样一个条件:

    • count(k) = min(count(k - coin[0]), ..., count(k - coin[n - 1])) + 1; 
    • 并且在符合条件k - coin[i] >= 0 && k - coin[i] < k的情况下, 前面的公式才成立.

    因为k - coin[i] < k的缘故, 那么在求count(k)时, 必须满足count(i)(i <- [0, k-1])已知, 所以这里又涉及到回溯的问题.

    所以我们可以创建一个矩阵matrix[k + 1][coin.length + 1], 使matrix[0][j]全部初始化为0值, 而在matrix[i][coin.length]保存面值为i的最少硬币数.

    而且具体的过程如下:

    1     * k|coin  1   3   5   min
    2     * 0       0   0   0   0
    3     * 1       1   0   0   1
    4     * 2       2   0   0   2
    5     * 3       3   1   0   3, 1
    6     * 4       2   2   0   2, 2
    7     * 5       3   3   1   3, 3, 1
    8     * 6       2   2   2   2, 2, 2
    9     * ...

    最后, 具体的Java代码实现如下:

     1     public static int backTrackingCoin(int[] coins, int k) {//回溯法+动态规划
     2         if (coins == null || coins.length == 0 || k < 1) {
     3             return 0;
     4         }
     5         int[][] matrix = new int[k + 1][coins.length + 1];
     6         for (int i = 1; i <= k; i++) {
     7             for (int j = 0; j < coins.length; j++) {
     8                 int preK = i - coins[j];
     9                 if (preK > -1) {//只有在不小于0时, preK才能存在于数组matrix中, 才能够进行回溯.
    10                     matrix[i][j] = matrix[preK][coins.length] + 1;//面值i在进行回溯
    11                     if (matrix[i][coins.length] == 0 || matrix[i][j] < matrix[i][coins.length]) {//如果当前的硬币数目是最少的, 更新min列的最少硬币数目
    12                         matrix[i][coins.length] = matrix[i][j];
    13                     }
    14                 }
    15             }
    16         }
    17         return matrix[k][coins.length];
    18     }

    代码经过测试, 题目给出的测试用例全部通过!

  • 相关阅读:
    J2ME学习笔记之问题看法
    单词王(kingWord)
    TCP的粘包和拆包问题及解决办法(C#)
    vuex里mapState,mapGetters使用详解
    agelform formcreate 使 elementui form 组件更简单
    elementui 二次封装系列 button
    开源一套后台管理系统框架,远离996 记录
    客户端架构介绍
    简单的Unity学习项目,封装了一下简单、通用功能组件,适用于数据可视化展示
    Vue关于对象数组的双向数据绑定(props与watch)
  • 原文地址:https://www.cnblogs.com/littlepanpc/p/7857599.html
Copyright © 2020-2023  润新知