SICP Exercise 4.14

Exercise 4.14

1、Eva的map能够工作是应该的,因为Eva在求值器中定义了map。

2、那么Louis的map为什么不能工作呢?

首先,让我们来看一下map的基本使用方式(在scheme中,而不是在求值器中):

(map (lambda (x) (* 2 x)) '(1 2 3 4))
=> (2 4 6 8)
可以看出map函数的使用方式和其他作为基本过程安装的函数(如cons,cdr等)的区别是它接受一个函数作为参数。好的,假设我们按照louis的方式实现map,现在我们来看看当我们的求值器接受上面的求值器时,会发生什么:

1)根据求值器规则,我们首先求值操作符map和参数的值:

;;; M-Eval input:
map

;;; M-Eval value:
(primitive #<procedure:mmap>)

;;; M-Eval input:
(lambda (x) (* 2 x))

;;; M-Eval value:
(compound-procedure (x) ((* 2 x)) <procedure-env>)

;;; M-Eval input:
'(1 2 3 4)

;;; M-Eval value:
(1 2 3 4)

作为对比,我们来看一下它们在scheme中的值:

> map
#<procedure:mmap>
> (lambda (x) (* 2 x))
#<procedure>
> '(1 2 3 4)
(1 2 3 4)

2)在求值过程调用时,求值器看到(primitive #<procedure:mmap>),它会取出系统的map版本,即#<procedure:mmap>,然后把它应用在参数(compound-procedure ...)上,这时map就会报错:

procedure application: expected procedure, given: (procedure (x) ((* 2 x)) ……
看到了吧,系统函数map期待的是#<procedure>类型的过程,而我们给的是(procedure (x) ((* 2 x)) 这样的列表。这个列表是由我们的求值器中的函数产生的:

(define (make-procedure parameters body env)
  (list 'procedure parameters body env))
因为系统版本的map不知道如何处理我们的求值器产生的过程,因此系统版本的map不能正常工作。

3,如何正确实现map?

或许最好的理解方式是像Eva一样自己实现一个map:

;; Exercise 4.14

(define (eval-map exp env)
  (let ((proc (cadr exp))
        (alist (cddr exp)))
    (apply-in-underlying-scheme
     map 
     (lambda (x) (apply (eval proc env) (list x)))
     (list-of-values alist env))))

(put 'map eval-map)
测试结果:

;;; M-Eval input:
(map (lambda (x) (* x 2)) '(1 2 3))

;;; M-Eval value:
(2 4 6)
注意:这里实现的map只是最基本的map函数,它只能接受一个列表,因为我们的求值器处理任意多个的参数的情况,像下面这样的方式map不能工作:

(map + '(1 2 3) '(4 5 6))






  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
"SICP中文版"是指计算机科学经典教材《Structure and Interpretation of Computer Programs》(计算机程序的构造和解释)的中文翻译PDF版。这本教材由麻省理工学院的Harold Abelson和Gerald Jay Sussman等人编写,是计算机科学领域中一本重要的教材。 "SICP中文版PDF"提供了更方便的学习方式。无论是学生、程序员还是计算机科学爱好者,都可以在任何时候通过电子设备访问和学习这本教材。使用PDF格式的好处是可以在不同的平台上都能打开和阅读,而不受限于特定的操作系统或设备。 通过"SICP中文版PDF",读者可以学习计算机科学的基本原理和概念,如过程、数据抽象、递归、高阶函数、并发等。这本教材以Scheme语言为示例,帮助读者理解计算机程序的结构、设计和解释。通过逐步的案例和练习,读者可以锻炼解决问题和编写高质量代码的能力。 "SICP中文版PDF"也提供了沟通和讨论的平台。读者可以通过在线社群或论坛,与其他人分享学习心得、解答疑问和参与讨论。这为读者提供了一个学习交流的机会,促进了学习者之间的互动和共同成长。 总之,"SICP中文版PDF"是一本经典的计算机科学教材的中文翻译版本,使得更多的读者可以方便地学习和掌握其中的知识。无论是对于计算机科学专业的学生还是对计算机科学感兴趣的人,这本教材都是一本很好的参考书,并提供了丰富的实例和练习,让读者深入理解计算机程序的核心概念和设计原则。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值