2.77
练习 2.77 Louis Reasoner 试着去求值 (magnitude z)
,其中的z就是图2-24里的那个对象。令他吃惊的是,从apply-generic
出来的不是5而是一个错误信息,说没办法对类型(complex)
做操作magnitude
。他将这次交互的情况给Alyssa P. Hacker看,Alyssa说“问题出在没有为complex
数定义复数选择函数,而只是为polar
和rectangular
数定义了它们。你需要做的就是在complex
包里加入下面这些东西”:
(put 'real-part '(complex) real-part)
(put 'imag-part '(complex) imag-part)
(put 'magnitude '(complex) magnitude)
(put 'angle '(complex) angle)
请详细说明为什么这样做是可行的。作为一个例子,请考虑表达式(magnitude z)
的求值过程,其中z就是图2-24里展示的那个对象,请追踪一下这一求值过程中的所有函数调用。特别是看看apply-generic
被调用了几次?每次调用中分派的是哪个过程?
先来测试一下 apply
函数:
' expect 10
(apply + (list 1 2 3 4))
x10
定义一个 error
函数:
(define (error msg v)
msg
)
#<undef>
然后定义一个 apply-generic
函数:
(define (attach-tag type-tag contents)
(cons type-tag contents)
)
(define (type-tag datum)
(if (pair? datum)
(car datum)
(error (string-append "Bad tagged datum -- TYPE-TAG" datum))
)
)
(define (contents datum)
(if (pair? datum)
(cdr datum)
(error "Bad tagged datum -- CONTENTS" datum)
)
)
(define (rectangular? z)
(eq? (type-tag z) 'rectangular)
)
(define (polar? z)
(eq? (type-tag z) 'polar)
)
(define (apply-generic op . args)
(let ((type-tags (map type-tag args)))
(let ((proc (get op (car type-tags))))
(if proc
(apply proc (map contents args))
(error
"No method for these types -- APPLY-GENERIC"
(list op type-tags)
)
)
)
)
)
(define (real-part z) (apply-generic 'real-part z))
(define (imag-part z) (apply-generic 'imag-part z))
(define (magnitude z) (apply-generic 'magnitude z))
(define (angle z) (apply-generic 'angle z))
#<undef>
尝试求 (magnitude z)
:
(magnitude z)
Error: execute: unbound symbol: "z" []true
z 还不存在。如何表示一个复数?先定义一个 make-complex-from-real-imag
函数:
(define (make-complex-from-real-imag x y)
((get 'make-from-real-imag 'complex) x y)
)
(define z (make-complex-from-real-imag 3 4))
Error: execute: unbound symbol: "get" [make-complex-from-real-imag]true
以上程序并不能直接执行,要能够执行,需要首先安装相应的程序包。要能安装包,需要先实现 get
和 put
函数。
(define helper (js-eval "const table = {}; function get (op, type) {console.log('getting ', op, type); return table[op][type];} function put (op,type, proc) { table[op] = table[op] || {}; table[op][type] = proc; } const exports = {get, put, table};
exports;"))
(define JSON (js-eval "JSON"))
#<undef>
(define (get op type)
(js-invoke helper 'get op type)
)
(define (put op type proc)
(js-invoke helper 'put op type proc)
)
#<undef>
有了 get
和 put
的支持,就可以写 install-package
函数了:
(define (square x) (* x x))
(define (install-rectangular-package)
;; internal procedures
(define (real-part z) (car z))
(define (imag-part z) (cdr z))
(define (make-from-real-imag x y) (cons x y))
(define (magnitude z)
(sqrt
(+
(square (real-part z))
(square (imag-part z))
)
)
)
(define (angle z)
(atan
(imag-part z)
(real-part z)
)
)
(define (make-from-mag-ang r a)
(cons
(* r (cos a))
(* r (sin a))
)
)
;; interface to the rest of the system
(define (tag x) (attach-tag 'rectangular x))
(put 'real-part 'rectangular real-part)
(put 'imag-part 'rectangular imag-part)
(put 'magnitude 'rectangular magnitude)
(put 'angle 'rectangular angle)
(put 'make-from-real-imag 'rectangular
(lambda (x y)
(tag
(make-from-real-imag x y)
)
)
)
(put 'make-from-mag-ang 'rectangular
(lambda (r a)
(tag
(make-from-mag-ang r a)
)
)
)
'done
)
(install-rectangular-package)
'done
(define (make-from-real-imag x y)
((get 'make-from-real-imag 'rectangular) x y)
)
(define z (make-from-real-imag 3 4))
z
('rectangular 3 . 4)
可以看到 z 现在是 ('rectangular 3 . 4)
的结构了。
这时,再求 (magnitude z)
。它会先求值: (type-tag z)
:
(type-tag z)
'rectangular
接着,再求 (contents z)
:
(contents z)
(3 . 4)
(magnitude z)
5
虽然得到了正确的结果,但这个 z 还只有 rectangular 一个标志。要成为如图 2-24 的结构,需要重新构造:
(define (install-complex-package)
;; internal procedures
;; interface to the rest of the system
(put 'real-part 'complex real-part)
(put 'imag-part 'complex imag-part)
(put 'magnitude 'complex magnitude)
(put 'angle 'complex angle)
(define (tag x) (attach-tag 'complex x))
(put 'make-complex-from-real-imag 'complex
(lambda (x y)
(tag
(make-from-real-imag x y)
)
)
)
'done
)
(install-complex-package)
(define (make-complex-from-real-imag x y)
((get 'make-complex-from-real-imag 'complex) x y)
)
(define z (make-complex-from-real-imag 3 4))
z
('complex 'rectangular 3 . 4)
现在 z 已经是如图 2-44 所示的结构了。再求:
(type-tag z)
'complex
(contents z)
('rectangular 3 . 4)
现在推测一下 real-part 的求值过程:
(real-part (list 'complex (list 'rectangular 3 4)))
(apply-generic 'real-part (list 'complex (list 'rectangular 3 4)))
((get 'real-part 'complex) (list 'rectangular 3 4))
(real-part (list 'rectangular 3 4))
(apply-generic 'real-part (list 'rectangular 3 4))
((get 'real-part 'rectangular) (list 3 4))
(real-part (list 3 4))
(car (list 3 4))
3
(real-part z)
3
果然是 3。最后求 (magnitude z) 结果如下:
(magnitude z)
5
这个层层解套的过程比较琐碎,暂时略过。大体如 (real-part z)。
其中 apply-generic
被调用了 2 次。第一次调用分派的是 (put 'magnitude 'complex magnitude)
注入的针对 complex
类型的 magnitude
;第二次调用分派的是 (put 'magnitude 'rectangular magnitude)
注入的针对 rectangular
类型的 magnitude
。