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

参考: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 = amount
    
    if amount < 1:
        ret_min = 0
    # 如果要找的钱数恰好是某种硬币的面值,那么最少只需一个硬币
    elif amount in values:
        ret_min = 1
    else:
        # 遍历面值数组中面值小于等于amount的那些元素
        for v in [x for x in values if x <= amount]:
            # 用面值为v的硬币+其他硬币找零所需的最少硬币数
            min_num = 1 + minCoins(amount - v)
            # 判断min_num和ret_min的大小,更新ret_min
            if min_num < ret_min:
                ret_min = min_num
                
    return ret_min

def 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 = amount
    
    if amount < 1:
        ret_min = 0
    # 如果要找的钱数恰好是某种硬币的面值,那么最少只需一个硬币
    elif amount in values:
        ret_min = 1
    else:
        # 遍历面值数组中面值小于等于amount的那些元素
        for v in [x for x in values if x <= amount]:
            # 用面值为v的硬币+其他硬币找零所需的最少硬币数
            min_num = 1 + minCoins(amount - v)
            # 判断min_num和ret_min的大小,更新ret_min
            if min_num < ret_min:
                ret_min = min_num
    
    recursion_num += 1
    return ret_min

def main():
    print minCoins(63)
    print recursion_num
    
main()  

将上面脚本保存成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),居然递归调用了函数minCoins67716925次,平均求每个值调用了1058076次。那么问题出在哪里了呢?出在了重复计算上,有很多值被重复计算了上百万次。那么如何尽量减少重复计算呢?下面用一个缓存数组来缓存每次求出的函数值,供后面使用,从而减少重复计算。

3. 性能优化版

# coding:utf-8
# 找零钱问题算法实现:基本版性能分析

# 统计递归次数
recursion_num = 0

# 4种硬币面值
values = [1,5,10,25]

# 缓存数组,为一个一维数组,用于缓存每次递归函数求得的值
# cache[i]表示凑够i美分所需的最少硬币个数,cache的元素都被初始化为-1,表示个数未知
cache = []

# 初始化缓存数组
def init(amount):
    global cache
    cache = [-1] * (amount + 1)

# 凑够amount这么多钱数需要的最少硬币个数
def minCoins(amount):
    global recursion_num
    global cache
    
    # 需要的最少硬币个数
    ret_min = amount
    
    # 如果缓存数组中有对应的值,那么直接从中取,不再重复计算了
    if cache[amount] != -1:
        ret_min = cache[amount]
    elif amount < 1:
        ret_min = 0
    # 如果要找的钱数恰好是某种硬币的面值,那么最少只需一个硬币
    elif amount in values:
        ret_min = 1
    else:
        # 遍历面值数组中面值小于等于amount的那些元素
        for v in [x for x in values if x <= amount]:
            # 用面值为v的硬币+其他硬币找零所需的最少硬币数
            min_num = 1 + minCoins(amount - v)
            # 判断min_num和ret_min的大小,更新ret_min
            if min_num < ret_min:
                ret_min = min_num
    
    # 更新缓存数组
    cache[amount] = ret_min
    
    recursion_num += 1
    return ret_min

def main():
    init(63)
    print minCoins(63)
    print cache
    print recursion_num
    
main()  

将上面脚本保存成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倍,优化效果是巨大的。

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

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,723评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,485评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,998评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,323评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,355评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,079评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,389评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,019评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,519评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,971评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,100评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,738评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,293评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,289评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,517评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,547评论 2 354
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,834评论 2 345

推荐阅读更多精彩内容

  • 动态规划(Dynamic Programming) 本文包括: 动态规划定义 状态转移方程 动态规划算法步骤 最长...
    廖少少阅读 3,256评论 0 18
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,515评论 25 707
  • 本文翻译自TopCoder上的一篇文章: Dynamic Programming: From novice to ...
    扎Zn了老Fe阅读 1,920评论 0 3
  • 像蚕一样 吐丝结茧 希望进不来 绝望出不去 只能自己品味忧伤
    雨墨蓝心阅读 236评论 0 2
  • 若桐像往常一样,回到家就写作业。我有些耐不往性子,走到若桐身边问“宝贝,你没有什么东西要给妈妈吗?”若桐楞住了。我...
    晚巷清风阅读 341评论 0 3