$动态规划系列(2)——找零钱问题

refer:http://interactivepython.org/courselib/static/pythonds/index.html

1. 问题描述

Tom在自动售货机上买了一瓶饮料,售价37美分,他投入了1美元(1美元 = 100美分),现在自动售货机需要找钱给他。售货机中现在只有四种面额的硬币:1美分、5美分、10美分、25美分,每种硬币的数量充足。现在要求使用最少数量的硬币,给Tom找钱,求出这个最少数量是多少。

2. 问题分析

自动售卖机需要给Tom找零钱63美分,而售卖机中只有四种面额的硬币可以使用,现在的核心问题就是如何用四种面额的硬币来凑够63美分,并且使用的硬币数量最少。

现在我们换个角度来思考这个问题:
是不是可以将问题规模先缩小?比如我不知道凑够63美分最少需要多少个硬币,那凑够1美分、2美分的方案则显而易见是可以马上知道的。
为了后面叙述方便,用f(i) = n这个等式来表示这样一种含义:凑够i美分(0 <= i <= 63)所需要的最少硬币数量为n个,那么我们从凑够0美分开始写:

  • 凑0美分:因为0美分根本不需要硬币,因此结果是0:f(0) = 0;

  • 凑1美分:因为有1美分面值的硬币可以使用,所以可以先用一个1美分硬币,然后再凑够0美分即可,而f(0)的值是我们已经算出来了的,所以:f(1) = 1 + f(0) = 1 + 0 = 1,这里f(1) = 1 + f(0) 中的1表示用一枚1美分的硬币;

  • 凑2美分:此时四种面额的硬币中只有1美分比2美分小,所以只能先用一个1美分硬币,然后再凑够1美分即可,而f(1)的值我们也已经算出来了,所以:f(2) = 1 + f(1) = 1 + 1 = 2,这里f(2) = 1 + f(1) 中的1表示用一枚1美分的硬币;

  • 凑3美分:和上一步同样的道理,f(3) = 1 + f(2) = 1 + 2 = 3;

  • 凑4美分:和上一步同样的道理,f(4) = 1 + f(3) = 1 + 3 = 4;

  • 凑5美分:这时就出现了不止一种选择了,因为有5美分面值的硬币。方案一:使用一个5美分的硬币,再凑够0美分即可,这时:f(5) = 1 + f(0) = 1 + 0 = 1,这里f(5) = 1 + f(0) 中的1表示用一枚5美分的硬币;方案二:使用1个1美分的硬币,然后再凑够4美分,此时:f(5) = 1 + f(4) = 1 + 4 = 5。综合方案一和方案二,可得:f(5) = min{1 + f(0),1 + f(4)} = 1;

  • 凑6美分:此时也有两种方案可选,方案一:先用一个1美分,然后再凑够5美分即可,即:f(6) = 1 + f(5) = 1 + 1 = 2;方案二:先用一个5美分,然后再凑够1美分即可,即:f(6) = 1 + f(1) = 1 + 1 = 2。综合两种方案,有:f(6) = min{1 + f(5), 1 + f(1)} = 2;

  • ...(省略)

从上面的分析过程可以看出,要凑够i美分,就要考虑如下各种方案的最小值:

1 + f(i - value[j]),其中value[j]表示第j种(j从0开始,0 <= j < 4)面值且value[j] <= i

那么现在就可以写出状态转移方程了:

f(i) = 0, i = 0

f(i) = 1, i = 1

f(i) = min{1 + f(i - value[j])}, i > 1,value[j] <= i

3. Talk is cheap, show the code

1. 基本版

# coding:utf-8
# 找零钱问题算法实现:基本版# 4种硬币面值
values = [1,5,10,25]# 凑够amount这么多钱数需要的最少硬币个数
def minCoins(amount):# 需要的最少硬币个数ret_min = amountif amount < 1:ret_min = 0# 如果要找的钱数恰好是某种硬币的面值,那么最少只需一个硬币elif amount in values:ret_min = 1else:# 遍历面值数组中面值小于等于amount的那些元素for v in [x for x in values if x <= amount]:# 用面值为v的硬币+其他硬币找零所需的最少硬币数min_num = 1 + minCoins(amount - v)# 判断min_num和ret_min的大小,更新ret_minif min_num < ret_min:ret_min = min_numreturn ret_mindef main():print minCoins(63)main()  

将上面脚本保存成coins.py文件,在ipython中执行:%time %run coins.py,得到的结果如下:

6

CPU times: user 1min 45s, sys: 0 ns, total: 1min 45s

Wall time: 1min 45s

分析:可以看出,在我的电脑上,仅仅是为了计算用4种面额找63美分零钱,就耗时1分钟45秒(105秒),这是无法忍受的。那么究竟为什么耗时这么巨大?下面对代码稍加改造进行一下性能分析。

2. 性能分析

# coding:utf-8
# 找零钱问题算法实现:基本版性能分析# 统计递归次数
recursion_num = 0# 4种硬币面值
values = [1,5,10,25]# 凑够amount这么多钱数需要的最少硬币个数
def minCoins(amount):global recursion_num# 需要的最少硬币个数ret_min = amountif amount < 1:ret_min = 0# 如果要找的钱数恰好是某种硬币的面值,那么最少只需一个硬币elif amount in values:ret_min = 1else:# 遍历面值数组中面值小于等于amount的那些元素for v in [x for x in values if x <= amount]:# 用面值为v的硬币+其他硬币找零所需的最少硬币数min_num = 1 + minCoins(amount - v)# 判断min_num和ret_min的大小,更新ret_minif min_num < ret_min:ret_min = min_numrecursion_num += 1return ret_mindef main():print minCoins(63)print recursion_nummain()  

将上面脚本保存成coins.py文件,在ipython中执行:%time %run coins.py,得到的结果如下:

6

67716925

CPU times: user 2min, sys: 36 ms, total: 2min

Wall time: 2min

分析:可见,minCoins函数一共被递归调用了67716925次,真是难以想象,为了计算最多64个函数值(amount取0~63),居然递归调用了函数minCoins 67716925次,平均求每个值调用了1058076次。那么问题出在哪里了呢?出在了重复计算上,有很多值被重复计算了上百万次。那么如何尽量减少重复计算呢?下面用一个缓存数组来缓存每次求出的函数值,供后面使用,从而减少重复计算。

3. 性能优化版

# coding:utf-8
# 找零钱问题算法实现:基本版性能分析# 统计递归次数
recursion_num = 0# 4种硬币面值
values = [1,5,10,25]# 缓存数组,为一个一维数组,用于缓存每次递归函数求得的值
# cache[i]表示凑够i美分所需的最少硬币个数,cache的元素都被初始化为-1,表示个数未知
cache = []# 初始化缓存数组
def init(amount):global cachecache = [-1] * (amount + 1)# 凑够amount这么多钱数需要的最少硬币个数
def minCoins(amount):global recursion_numglobal cache# 需要的最少硬币个数ret_min = amount# 如果缓存数组中有对应的值,那么直接从中取,不再重复计算了if cache[amount] != -1:ret_min = cache[amount]elif amount < 1:ret_min = 0# 如果要找的钱数恰好是某种硬币的面值,那么最少只需一个硬币elif amount in values:ret_min = 1else:# 遍历面值数组中面值小于等于amount的那些元素for v in [x for x in values if x <= amount]:# 用面值为v的硬币+其他硬币找零所需的最少硬币数min_num = 1 + minCoins(amount - v)# 判断min_num和ret_min的大小,更新ret_minif min_num < ret_min:ret_min = min_num# 更新缓存数组cache[amount] = ret_minrecursion_num += 1return ret_mindef main():init(63)print minCoins(63)print cacheprint recursion_nummain()  

将上面脚本保存成coins.py文件,在ipython中执行:%time %run coins.py,得到的结果如下:

6

[-1, 1, 2, 3, 4, 1, 2, 3, 4, 5, 1, 2, 3, 4, 5, 2, 3, 4, 5, 6, 2, 3, 4, 5, 6, 1, 2, 3, 4, 5, 2, 3, 4, 5, 6, 2, 3, 4, 5, 6, 3, 4, 5, 6, 7, 3, 4, 5, 6, 7, 2, 3, 4, 5, 6, 3, 4, 5, 6, 7, 3, 4, 5, 6]

206

CPU times: user 4 ms, sys: 0 ns, total: 4 ms

Wall time: 2.2 ms

分析:可见,cache数组除了cache[0]没被用到以外,其他元素都被利用到了,利用率还是很高的。使用缓存数组后,minCoins函数的递归调用次数从67716925次降低到了206次,降低了328722倍;程序耗时从105秒降低到了2.2ms,降低了47727倍,优化效果是巨大的。

上一篇动态规划之金矿模型中也使用到了缓存数组,优化效果也是巨大的,在本文中又一次看到了动态规划中缓存数组的重要性。

转载于:https://www.cnblogs.com/jiayongji/p/7118895.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/416460.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

前端学习(2467):在前端页面中引入百度地图

走在前端的大道上 插槽&#xff0c;也就是slot&#xff0c;是组件的一块HTML模板&#xff0c;这块模板显示不显示、以及怎样显示由父组件来决定。 实际上&#xff0c;一个slot最核心的两个问题在这里就点出来了&#xff0c;是显示不显示和怎样显示。 由于插槽是一块模板&…

pm961 mysql_Oracle GoldenGate学习之--异构平台同步(Mysql到Oracle)

Oracle GoldenGate学习之--异构平台同步(Mysql到Oracle)如图所示&#xff1a;源端采用Mysql库&#xff0c;目标端采用Oracle库一、OGG安装配置(源端)1、OGG下载https://edelivery.oracle.com/EPD/Download/get_form?egroup_aru_number14841438https://edelivery.oracle.com/EP…

错误记录

1.Soap 1.1 endpoint already registered on address /cxfService答案&#xff1a;web.xml中多配置了 <init-param> <param-name>config-location</param-name> <param-value>classpath:cxf.xml</param-value> </init-param> 2.The serve…

sqlalchemy mysql配置中怎么设置utf8_在SqlAlchemy中,我想要一个列是UTF8?

我注意到我的MySQL数据库在默认情况下没有设置为UTF-8。改为选择latin1_swedish_ci排序规则。在所以我很自然地遇到了一个用户报告的错误&#xff0c;我的应用程序不支持特殊字符。我去确保我的应用程序正确地处理了UTF-8&#xff0c;编写了一个测试&#xff0c;很肯定它可以很…

ado.net mysql 连接池_ADO.NET数据库连接池的介绍 | 学步园

摘录自MSDN&#xff1a;建立池连接可以显著提高应用程序的性能和可缩放性。SQL Server .NET Framework 数据提供程序自动为 ADO.NET客户端应用程序提供连接池。您也可以提供几个连接字符串修饰符来控制连接池行为&#xff0c;请参见本主题内下文中“使用连接字符串关键字控制连…

转:在csv中维护变量参数

问题&#xff1a; 1、我的变量表多&#xff0c;通过之前的csv获取的方式&#xff0c;或者用户变量来维护&#xff0c;比较麻烦 2、我想在脚本之外维护我的变量数据&#xff0c;脱离脚本 解决方案&#xff1a; 1、csv的配置如图&#xff0c;队列是变量名称&#xff0c;第二列是变…

前端学习(2471):vue-echarts和echarts的区别:

vue-echarts和echarts的区别&#xff1a; vue-echarts是封装后的vue插件&#xff0c; 基于 ECharts v4.0.1 开发&#xff0c;依赖 Vue.js v2.2.6&#xff0c;功能一样的只是把它封装成vue插件 这样更方便以vue的方式去使用它。echarts就是普通的js库&#xff0c; vue-echarts特…

python 零代码快速开发平台_现在低代码开发平台和零代码平台区别是什么?

低代码与零代码&#xff0c;可参考一下开发平台在国内已经发展了很久了&#xff0c;从有代码到低代码&#xff0c;甚至有些厂家声称可以无代码&#xff0c;当然无代码只是一个噱头。低代码开发可以解决传统的开发方式的复杂方式&#xff0c;提高开发效率&#xff0c;节省时间成…

JSON.parse()、eval()和JSON.stringify()

1.什么是JSON&#xff1f; JSON全称是JavaScript Object Notation,是一种轻量级的数据交换格式。JSON 与XML具有相同的特性&#xff0c;是一种数据存储格式&#xff0c;但是JSON相比XML 更易于人编写和阅读&#xff0c;更易于生成和解析。 2.json对象和json字符串 json对象可以…

rust edition python3_PyO3首页、文档和下载 - Python 解释器的 Rust 绑定

PyO3 是 Python 的 Rust 绑定&#xff0c;可以用 Rust 语言对 Python 加速。这包括用 Rust 语言运行 Python 代码并与之交互&#xff0c;以及直接编写原生 Python 模块。PyO3 一开始只是作为 rust-cpython 的分支出现, 后来由于 rust-cpython 缺乏维护, PyO3 开始在 Rust 社区流…

solr 配置中文分词器

ik转载于:https://www.cnblogs.com/javabigdata/p/7127639.html

前端学习(2474):页面布局

request.js <template> <div class"artical-container"><!--卡片--><el-card class"filter-card"><div slot"header" class"clearfix"><!--面包屑导航--><el-breadcrumb separator-class&quo…

https证书/即SSL数字证书申请途径和流程

国际CA机构GlobalSign中国 数字证书颁发中心网站&#xff1a;http://cn.globalsign.com      https证书即SSL数字证书&#xff0c;是广泛用 于网站通讯加密传输的解决方案&#xff0c;是提供通信保密的安全性协议&#xff0c;现已成为用来鉴别网站的真实身份&#xff0c;以…

linuxsed替换字符串后保存_Numpy运用-文件读写、存储及字符串处理

问题列举&#xff1a;Numpy文件读取Numpy文件存储Numpy字符串操作1、文件读取可以使用genfromtxt读取txt或者csv文件可以使用loadtxt读取txt或者csv文件两个函数功能类似&#xff0c;genfromtxt针对的更多是结构化数据注&#xff1a;delimiter表示的是以&#xff0c;分隔数据&a…