SICP 找零钱问题背后的思考

问题见SICP P26

此问题的递归方法很简单,类似于背包的思想。

即金额为amount的现金换成n种硬币的种类数 满足循环不变式:

count_change(amount,n)=count_change(amount,n-1)+count_change(amount-amount_of_first_coin,n)

递归中止条件是:当a=0,结果为1

a<0,结果为0

当n=0 结果也为0

  • 将上述规则转换为scheme代码,在Drracket中运行
 1      #lang racket
 2      (define (count-change amount kinds-of-coins)
 3           ( cond ((= amount 0) 1)
 4          ((or (< amount 0) (= kinds-of-coins 0))0)
 5          (else
 6            (+(count-change amount (- kinds-of-coins 1)) (count-change (- amount (some-coin kinds-of-coins)) kinds-of-coins )) ) ) )
 7
 8      (define (some-coin kinds-of-coins)
 9        ( cond ((= kinds-of-coins 1) 1)
10          ((= kinds-of-coins 2) 5)
11          ((= kinds-of-coins 3) 10)
12          ((= kinds-of-coins 4) 25)
13          ((= kinds-of-coins 5) 50)))
14
15   (count-change 45 5)
  • 上述代码在解决amount=300以上的时候已经十分缓慢了,原因在于这是个递归,并非尾递归(迭代),有大量重复和冗余的计算在其中,但此问题比斐波那契数复杂,因为

斐波那契数的问题,我们容易将其转化为迭代,因为此问题性质很好,每次分支只有2,每个问题直接满足最优自问题性质。但换零钱问题则不然,无法简单的化为尾递归,那么除了将递归转化为迭代外,另一个折衷的优化技巧是动态规划。

思路如下:

零钱coin=[c1,c2,c3……]

【 amount金额的钱币换成 kind_of_coins种零钱的种类数】=【只有一种零钱c1换的种类数】+【用两种零钱c1 c2的种类数】+……

可以看到,【用两种零钱c1 c2的种类数】就是在【只有一种零钱c1换的种类数】的基础上对第二个零钱c2做同样处理

 1  //cpp
 2           #include <iostream>
 3           #include <vector>
 4           using namespace std;
 5
 6           int main ()
 7           {
 8                  int amount = 55;
 9                  const int kind_of_coins = 5 ;
10                  int coin [ kind_of_coins] = { 1 , 5 , 10, 25 , 50 };
11                  vector< int > result ( amount + 1 , 0 );
12                  result [0 ] = 1;
13
14                  for (int i = 0 ; i < kind_of_coins; ++ i ){
15                        int j = coin[ i ];
16                        for (; j <= amount; ++ j )
17                              result [j ] += result[ j - coin [ i]];
18                   }
19
20               cout << result [amount ] << endl;
21               system ("pause" );
22
23           }
24  

改用python:

 1  //cpp
 2           #include <iostream>
 3           #include <vector>
 4           using namespace std;
 5
 6           int main ()
 7           {
 8                  int amount = 55;
 9                  const int kind_of_coins = 5 ;
10                  int coin [ kind_of_coins] = { 1 , 5 , 10, 25 , 50 };
11                  vector< int > result ( amount + 1 , 0 );
12                  result [0 ] = 1;
13
14                  for (int i = 0 ; i < kind_of_coins; ++ i ){
15                        int j = coin[ i ];
16                        for (; j <= amount; ++ j )
17                              result [j ] += result[ j - coin [ i]];
18                   }
19
20               cout << result [amount ] << endl;
21               system ("pause" );
22
23           }

递归和迭代是一个从编程语言入门开始即有的问题,然后会不断重复感觉理解了,又发现新的内容,又感觉理解了的过程。

不论是c++还是python,语法上给出的都是循环结构,而显示的循环结构实质上是尾递归模式,就是迭代的一种刻画,而尾递归要比循环更容易体现迭代的内涵,

但是,这些c 家族的语言都不支持尾递归,所以你写成尾递归,有两种可能:第一种,编译器把它优化为循环;第二种,编译器把它作为普通递归来处理,冗余的计算很多。

因此,我们一般都将递归优化为循环。

容易让我们产生错觉,循环是高效的,而递归是低效的。

让我们看看scheme,lisp的一种方言,语法中支持尾递归来实现迭代,对于斐波那契数生成的两个版本:

#lang racket

递归版本
(define (fibonacci n)
  (cond( (= n 0) 0)
     ( (= n 1) 1)
    (else (+ (fibonacci (- n 1)) (fibonacci (- n 2))))))

迭代版本

#lang racket
(define (fibonacci n)
  (fib 0 1 n))
(define (fib first-number second-number n)
  (if   (= n 0)

    first-number
    (fib second-number (+ first-number second-number) (- n 1) )))

容易发现,迭代比递归快的多,因为它没有多余的重复计算,而且更重要的是它每次维护常数空间,而无须先扩展再收缩,并且还要记住运算的轨迹。

而一般的优化方法除了改为迭代,还有人工模拟栈或者动态规划,栈模拟没什么好说的,将递归栈人为构建。

而动态规划呢,其实很简单,就是我们对递归过程不能转化为尾递归即迭代的情况下,人为的记录下子过程的返回值,计算大过程的时候就可以直接范围,缺点在于需要维护一个比较大的空间,是典型的空间换时间,不过这是值得的,存储一定程度上的廉价的,而时间效率更加宝贵(当然有一定限度,若是存储无限算法意义变得微小)。

SICP 找零钱问题背后的思考

时间: 2024-10-02 02:52:09

SICP 找零钱问题背后的思考的相关文章

动态规划法(二)找零钱问题

??本次博客尝试以storyline的方式来写作,如有不足之处,还请多多包涵~~ 问题的诞生 ??我们故事的主人公叫做丁丁,他是一个十几岁的小男孩,机智聪颖,是某某杂货店的小学徒.在他生活的国度里,只流通面额为1,3,4的硬币.复杂这家店的店长,叫做老王,是个勤奋实干的中年人,每天都要跟钱打交道. ??有一天,他心血来潮,叫住正在摆放货物的丁丁,对他说道:"丁丁,你不是学过计算机方面的算法吗?我这里正好有个问题,不知你能解答不?" ??一听到算法,丁丁的眼睛里闪出光芒,这正是自己的兴趣

AOJ 169 找零钱 DP OR 母函数

一直觉得这题因为有总量限制,是不能用母函数解的,今天偶然发现原来是可以的,记录一下. 只要搞母函数的时候多开一维来表示用了多少个硬币就好了,其实就是目标状态是二维的母函数 类似于 假设我现在要处理的面值是2      (1 + x^2 * y + x^4 * y ^ 2 + x ^ 6 * y ^ 3...) 就表示用0个,1个,2个,3个..硬币的状态了. 看来母函数最重要的还是对式子本身的理解,这样才能应对各种变化. #include <cstdio> #include <cstri

贪婪算法_找零钱

贪婪算法是一种求近似解的方法,它存在如下几个问题: 1.不能保证最后的解是最优解. 2.不能求最大解或者最小解问题. 3.只能满足某些约束条件的可行解范围. 下面给出用贪婪算法解决找零钱这一问题的代码: 1 #include<stdio.h> 2 #define max 7 3 float facevalue[max]={50,20,10,5,1,0.5,0.1};//同时是否有无该面值用于找零,也能在此处进行修改 4 int num[max]={0}; 5 float exchange(fl

PAT 乙级 1037 在霍格沃茨找零钱(20)C++版

1037. 在霍格沃茨找零钱(20) 时间限制 400 ms 内存限制 65536 kB 代码长度限制 8000 B 判题程序 Standard 作者 CHEN, Yue 如果你是哈利·波特迷,你会知道魔法世界有它自己的货币系统 -- 就如海格告诉哈利的:"十七个银西可(Sickle)兑一个加隆(Galleon),二十九个纳特(Knut)兑一个西可,很容易."现在,给定哈利应付的价钱P和他实付的钱A,你的任务是写一个程序来计算他应该被找的零钱. 输入格式: 输入在1行中分别给出P和A,

1037. 在霍格沃茨找零钱(20)

1037. 在霍格沃茨找零钱(20) 如果你是哈利·波特迷,你会知道魔法世界有它自己的货币系统 -- 就如海格告诉哈利的:"十七个银西可(Sickle)兑一个加隆(Galleon),二十九个纳特(Knut)兑一个西可,很容易."现在,给定哈利应付的价钱P和他实付的钱A,你的任务是写一个程序来计算他应该被找的零钱. 输入格式: 输入在1行中分别给出P和A,格式为"Galleon.Sickle.Knut",其间用1个空格分隔.这里Galleon是[0, 107]区间内的

1028: 在霍格沃茨找零钱

1028: 在霍格沃茨找零钱 时间限制: 1 Sec  内存限制: 128 MB提交: 316  解决: 147[提交][状态][讨论版] 题目描述 如果你是哈利·波特迷,你会知道魔法世界有它自己的货币系统 —— 就如海格告诉哈利的:“十七个银西可(Sickle)兑一个加隆(Galleon),二十九个纳特(Knut)兑一个西可,很容易.”现在,给定哈利应付的价钱P和他实付的钱A,你的任务是写一个程序来计算他应该被找的零钱. 输入 输入在1行中分别给出P和A,格式为“Galleon.Sickle.

c语言趣题之“找零钱的方法数量 ”

/* Name: Copyright: Author: Date: 31-12-14 16:51 Description: 找零钱的方法数量 描述 我们知道人民币有1.2.5.10.20.50.100这几种面值. 现在给你n(1≤n≤250)元,让你计算换成用上面这些面额表示且总数不超过100张,共有几种. 比如4元,能用4张1元.2张1元和1张2元.2张2元,三种表示方法. 输入 输入有多组,每组一行,为一个整合n. 输入以0结束. 输出 输出该面额有几种表示方法. 样例输入 1 4 0 样例

贪心算法-找零钱(C#实现)

找零钱这个问题很清楚,无非就是始终拿可以取的最大面值来找,最后就使得张数最小了,这个实现是在假设各种面值足够多的情况下. 首先拖出一个界面来,最下面是一个listbox控件 对应的代码:问题比较简单,有注释 using System;using System.Collections.Generic;using System.ComponentModel;using System.Data;using System.Drawing;using System.Linq;using System.Te

翻译:动态规划--找零钱 coin change

来自http://www.geeksforgeeks.org/dynamic-programming-set-7-coin-change/ 对于整数N,找出N的所有零钱的表示.零钱可以用S={s1,s2,s3,..sm}表示,每种零钱的数量为无穷.请问有多少种找零的方法? 例如, N = 4,S = {1,2,3},有四种找零方式{1,1,1,1},{1,1,2},{2,2},{1,3},return 4 N = 10,S= {2,5,3,6} ,有5中找零方式{2,2,2,2,2}, {2,2