Java动态规划之硬币找零问题实现代码

作者:SilentKnight 时间:2023-01-23 20:37:38 

动态规划的基本思想是将待求解问题分解成若干个子问题,先求解子问题,并将这些子问题的解保存起来,如果以后在求解较大子问题的时候需要用到这些子问题的解,就可以直接取出这些已经计算过的解而免去重复运算。保存子问题的解可以使用填表方式,例如保存在数组中。

用一个实际例子来体现动态规划的算法思想——硬币找零问题。

问题描述:

假设有几种硬币,并且数量无限。请找出能够组成某个数目的找零所使用最少的硬币数。例如几种硬币为[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的最少硬币数.

而且具体的过程如下:


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

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


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

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

来源:http://www.cnblogs.com/littlepanpc/p/7857599.html

标签:java,算法,硬币找零,动态规划
0
投稿

猜你喜欢

  • Android自定义实现可滑动按钮

    2021-10-21 08:24:11
  • java8中的默认垃圾回收器(GC)

    2021-12-01 04:27:30
  • android studio2.3如何编译动态库的过程详解

    2023-07-11 03:47:48
  • 通过实例解析JMM和Volatile底层原理

    2023-05-20 19:10:48
  • java Iterator接口和LIstIterator接口分析

    2023-05-23 21:31:24
  • C#中倒序输出字符串的方法示例

    2023-10-27 21:45:13
  • SpringBoot通过自定义注解实现参数校验

    2023-09-21 21:11:02
  • Java中获取泛型类型信息的方法

    2022-06-30 16:06:34
  • C#读写指定编码格式的文本文件

    2023-06-16 03:45:41
  • Springboot整合Netty实现RPC服务器的示例代码

    2023-07-14 11:35:35
  • Android仿美团下拉菜单(商品选购)实例代码

    2023-05-07 06:03:34
  • Java网络编程之TCP通信完整代码示例

    2022-10-18 21:42:57
  • Kotlin基础通关之字符串与数字类型

    2023-06-22 13:27:35
  • spring boot 即时重新启动(热更替)使用说明

    2023-01-19 02:41:05
  • Java案例使用比较排序器comparator实现成绩排序

    2023-10-16 01:37:24
  • Mybatis配置错误:java.lang.ExceptionInInitializerError

    2021-12-31 16:58:59
  • 协定需要会话,但是绑定“BasicHttpBinding”不支持它或者因配置不正确而无法支持它

    2023-03-17 16:44:34
  • java中生产者消费者问题和代码案例

    2023-11-24 04:09:07
  • java8实现List中对象属性的去重方法

    2023-08-30 20:50:48
  • ADO.NET实用技巧两则

    2021-12-26 12:48:32
  • asp之家 软件编程 m.aspxhome.com