Finished 2.80
This commit is contained in:
parent
a72f7f9b7c
commit
b577186c70
3
2_74.rkt
3
2_74.rkt
|
@ -159,3 +159,6 @@
|
|||
;; To add a new division, it is necessary to put the corresponding constructors
|
||||
;; and selectors into the table with put, ensuring that each record and file is
|
||||
;; tagged.
|
||||
|
||||
;; Possible improvement: Only tag records in the get-record procedure. Then there
|
||||
;; is no need to change the record representation.
|
||||
|
|
|
@ -0,0 +1,21 @@
|
|||
#lang sicp
|
||||
|
||||
(define (make-from-mag-ang r theta)
|
||||
(lambda (op)
|
||||
(cond ((eq? op 'real-part) (* r (cos theta)))
|
||||
((eq? op 'imag-part) (* r (sin theta)))
|
||||
((eq? op 'magnitude) r)
|
||||
((eq? op 'angle) theta)
|
||||
(else (error "Unknown op: make-from-mag-ang" op)))))
|
||||
|
||||
;; Generic operations, explicit dispatch: to add a new operation, a new rule must be added
|
||||
;; for each existing type. To add a new type, a new rule must be added for each existing
|
||||
;; operation.
|
||||
;;
|
||||
;; Data directed: to add a new operation, a new entry is added to the table for each type.
|
||||
;; To add a new type, a new entry is added to the table for each operation.
|
||||
;;
|
||||
;; Message passing; to add a new operation a new conditional case is added to each relevant
|
||||
;; object. To add a new type, a new object is created with all required operations.
|
||||
|
||||
|
|
@ -0,0 +1,16 @@
|
|||
#lang sicp
|
||||
|
||||
|
||||
(define (attach-tag type-tag contents)
|
||||
(cond ((number? contents) contents)
|
||||
(else (cons type-tag contents))))
|
||||
|
||||
(define (type-tag datum)
|
||||
(cond ((number? datum) 'scheme-number)
|
||||
((pair? datum) (car datum))
|
||||
(else (error "Bad tagged datum -- TYPE-TAG" datum))))
|
||||
|
||||
(define (contents datum)
|
||||
(cond ((number? datum) datum)
|
||||
((pair? datum) (cdr datum))
|
||||
(else (error "Bad tagged datum -- CONTENTS" datum))))
|
|
@ -0,0 +1,288 @@
|
|||
#lang sicp
|
||||
|
||||
|
||||
;;;from chapter 1
|
||||
(define (square x) (* x x))
|
||||
|
||||
;;;from section 1.2.5, for Section 2.1.1
|
||||
(define (gcd a b)
|
||||
(if (= b 0)
|
||||
a
|
||||
(gcd b (remainder a b))))
|
||||
|
||||
;;;-----------
|
||||
;;;from section 3.3.3 for section 2.4.3
|
||||
;;; to support operation/type table for data-directed dispatch
|
||||
|
||||
(define (assoc key records)
|
||||
(cond ((null? records) false)
|
||||
((equal? key (caar records)) (car records))
|
||||
(else (assoc key (cdr records)))))
|
||||
|
||||
(define (make-table)
|
||||
(let ((local-table (list '*table*)))
|
||||
(define (lookup key-1 key-2)
|
||||
(let ((subtable (assoc key-1 (cdr local-table))))
|
||||
(if subtable
|
||||
(let ((record (assoc key-2 (cdr subtable))))
|
||||
(if record
|
||||
(cdr record)
|
||||
false))
|
||||
false)))
|
||||
(define (insert! key-1 key-2 value)
|
||||
(let ((subtable (assoc key-1 (cdr local-table))))
|
||||
(if subtable
|
||||
(let ((record (assoc key-2 (cdr subtable))))
|
||||
(if record
|
||||
(set-cdr! record value)
|
||||
(set-cdr! subtable
|
||||
(cons (cons key-2 value)
|
||||
(cdr subtable)))))
|
||||
(set-cdr! local-table
|
||||
(cons (list key-1
|
||||
(cons key-2 value))
|
||||
(cdr local-table)))))
|
||||
'ok)
|
||||
(define (dispatch m)
|
||||
(cond ((eq? m 'lookup-proc) lookup)
|
||||
((eq? m 'insert-proc!) insert!)
|
||||
(else (error "Unknown operation -- TABLE" m))))
|
||||
dispatch))
|
||||
|
||||
(define operation-table (make-table))
|
||||
(define get (operation-table 'lookup-proc))
|
||||
(define put (operation-table 'insert-proc!))
|
||||
|
||||
;; Apply generic
|
||||
|
||||
(define (apply-generic op . args)
|
||||
(let ((type-tags (map type-tag args)))
|
||||
(let ((proc (get op type-tags)))
|
||||
(if proc
|
||||
(apply proc (map contents args))
|
||||
(error
|
||||
"No method for these types -- APPLY-GENERIC"
|
||||
(list op type-tags))))))
|
||||
|
||||
;; Scheme numbers
|
||||
|
||||
(define (install-scheme-number-package)
|
||||
(define (tag x)
|
||||
(attach-tag 'scheme-number x))
|
||||
(put 'add '(scheme-number scheme-number)
|
||||
(lambda (x y) (tag (+ x y))))
|
||||
(put 'sub '(scheme-number scheme-number)
|
||||
(lambda (x y) (tag (- x y))))
|
||||
(put 'mul '(scheme-number scheme-number)
|
||||
(lambda (x y) (tag (* x y))))
|
||||
(put 'div '(scheme-number scheme-number)
|
||||
(lambda (x y) (tag (/ x y))))
|
||||
(put 'equ? '(scheme-number scheme-number) =)
|
||||
(put '=zero? '(scheme-number)
|
||||
(lambda (x) (= x 0)))
|
||||
(put 'make 'scheme-number
|
||||
(lambda (x) (tag x)))
|
||||
'done)
|
||||
|
||||
;; Rationals
|
||||
|
||||
(define (install-rational-package)
|
||||
;; internal procedures
|
||||
(define (numer x) (car x))
|
||||
(define (denom x) (cdr x))
|
||||
(define (make-rat n d)
|
||||
(let ((g (gcd n d)))
|
||||
(cons (/ n g) (/ d g))))
|
||||
(define (add-rat x y)
|
||||
(make-rat (+ (* (numer x) (denom y))
|
||||
(* (numer y) (denom x)))
|
||||
(* (denom x) (denom y))))
|
||||
(define (sub-rat x y)
|
||||
(make-rat (- (* (numer x) (denom y))
|
||||
(* (numer y) (denom x)))
|
||||
(* (denom x) (denom y))))
|
||||
(define (mul-rat x y)
|
||||
(make-rat (* (numer x) (numer y))
|
||||
(* (denom x) (denom y))))
|
||||
(define (div-rat x y)
|
||||
(make-rat (* (numer x) (denom y))
|
||||
(* (denom x) (numer y))))
|
||||
(define (eq-rat x y)
|
||||
(and (= (numer x) (numer y))
|
||||
(= (denom x) (denom y))))
|
||||
;; interface to rest of the system
|
||||
(define (tag x) (attach-tag 'rational x))
|
||||
(put 'add '(rational rational)
|
||||
(lambda (x y) (tag (add-rat x y))))
|
||||
(put 'sub '(rational rational)
|
||||
(lambda (x y) (tag (sub-rat x y))))
|
||||
(put 'mul '(rational rational)
|
||||
(lambda (x y) (tag (mul-rat x y))))
|
||||
(put 'div '(rational rational)
|
||||
(lambda (x y) (tag (div-rat x y))))
|
||||
(put 'equ? '(rational rational) eq-rat)
|
||||
(put '=zero? '(rational)
|
||||
(lambda (x) (= (numer x) 0)))
|
||||
|
||||
(put 'make 'rational
|
||||
(lambda (n d) (tag (make-rat n d))))
|
||||
'done)
|
||||
|
||||
;;;-----------
|
||||
;;;SECTION 2.4.3
|
||||
|
||||
;; uses get/put (from 3.3.3) -- see ch2support.scm
|
||||
|
||||
(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)
|
||||
|
||||
(define (install-polar-package)
|
||||
;; internal procedures
|
||||
(define (magnitude z) (car z))
|
||||
(define (angle z) (cdr z))
|
||||
(define (make-from-mag-ang r a) (cons r a))
|
||||
(define (real-part z)
|
||||
(* (magnitude z) (cos (angle z))))
|
||||
(define (imag-part z)
|
||||
(* (magnitude z) (sin (angle z))))
|
||||
(define (make-from-real-imag x y)
|
||||
(cons (sqrt (+ (square x) (square y)))
|
||||
(atan y x)))
|
||||
|
||||
;; interface to the rest of the system
|
||||
(define (tag x) (attach-tag 'polar x))
|
||||
(put 'real-part '(polar) real-part)
|
||||
(put 'imag-part '(polar) imag-part)
|
||||
(put 'magnitude '(polar) magnitude)
|
||||
(put 'angle '(polar) angle)
|
||||
(put 'make-from-real-imag 'polar
|
||||
(lambda (x y) (tag (make-from-real-imag x y))))
|
||||
(put 'make-from-mag-ang 'polar
|
||||
(lambda (r a) (tag (make-from-mag-ang r a))))
|
||||
'done)
|
||||
|
||||
|
||||
(define (install-complex-package)
|
||||
;; imported procedures from rectangular and polar packages
|
||||
(define (make-from-real-imag x y)
|
||||
((get 'make-from-real-imag 'rectangular) x y))
|
||||
(define (make-from-mag-ang r a)
|
||||
((get 'make-from-mag-ang 'polar) r a))
|
||||
;; internal procedures
|
||||
(define (add-complex z1 z2)
|
||||
(make-from-real-imag (+ (real-part z1) (real-part z2))
|
||||
(+ (imag-part z1) (imag-part z2))))
|
||||
(define (sub-complex z1 z2)
|
||||
(make-from-real-imag (- (real-part z1) (real-part z2))
|
||||
(- (imag-part z1) (imag-part z2))))
|
||||
(define (mul-complex z1 z2)
|
||||
(make-from-mag-ang (* (magnitude z1) (magnitude z2))
|
||||
(+ (angle z1) (angle z2))))
|
||||
(define (div-complex z1 z2)
|
||||
(make-from-mag-ang (/ (magnitude z1) (magnitude z2))
|
||||
(- (angle z1) (angle z2))))
|
||||
|
||||
;; interface to rest of the system
|
||||
(define (tag z) (attach-tag 'complex z))
|
||||
(put 'add '(complex complex)
|
||||
(lambda (z1 z2) (tag (add-complex z1 z2))))
|
||||
(put 'sub '(complex complex)
|
||||
(lambda (z1 z2) (tag (sub-complex z1 z2))))
|
||||
(put 'mul '(complex complex)
|
||||
(lambda (z1 z2) (tag (mul-complex z1 z2))))
|
||||
(put 'div '(complex complex)
|
||||
(lambda (z1 z2) (tag (div-complex z1 z2))))
|
||||
(put 'equ? '(complex complex)
|
||||
(lambda (z1 z2)
|
||||
(and (= (real-part z1) (real-part z2))
|
||||
(= (imag-part z1) (imag-part z2)))))
|
||||
(put '=zero? '(complex)
|
||||
(lambda (z) (= (magnitude z) 0)))
|
||||
(put 'make-from-real-imag 'complex
|
||||
(lambda (x y) (tag (make-from-real-imag x y))))
|
||||
(put 'make-from-mag-ang 'complex
|
||||
(lambda (r a) (tag (make-from-mag-ang r a))))
|
||||
'done)
|
||||
|
||||
|
||||
;; Generic constructors
|
||||
|
||||
(define (make-scheme-number n)
|
||||
((get 'make 'scheme-number) n))
|
||||
|
||||
(define (make-rational n d)
|
||||
((get 'make 'rational) n d))
|
||||
|
||||
(define (make-complex-from-real-imag x y)
|
||||
((get 'make-from-real-imag 'complex) x y))
|
||||
|
||||
(define (make-complex-from-mag-ang r a)
|
||||
((get 'make-from-mag-ang 'complex) r a))
|
||||
|
||||
;; Generic selectors
|
||||
|
||||
(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))
|
||||
|
||||
(put 'real-part '(complex) real-part)
|
||||
(put 'imag-part '(complex) imag-part)
|
||||
(put 'magnitude '(complex) magnitude)
|
||||
(put 'angle '(complex) angle)
|
||||
|
||||
;; Generic operators
|
||||
|
||||
(define (add x y) (apply-generic 'add x y))
|
||||
(define (sub x y) (apply-generic 'sub x y))
|
||||
(define (mul x y) (apply-generic 'mul x y))
|
||||
(define (div x y) (apply-generic 'div x y))
|
||||
|
||||
(define (equ? x y) (apply-generic 'equ? x y))
|
||||
;;(put 'equ? '(complex complex) equ?)
|
||||
|
||||
(define (=zero? x) (apply-generic '=zero? x))
|
||||
|
||||
(install-scheme-number-package)
|
||||
(install-rational-package)
|
||||
(install-polar-package)
|
||||
(install-rectangular-package)
|
||||
(install-complex-package)
|
||||
|
||||
|
||||
|
||||
(define (attach-tag type-tag contents)
|
||||
(cond ((number? contents) contents)
|
||||
(else (cons type-tag contents))))
|
||||
|
||||
(define (type-tag datum)
|
||||
(cond ((number? datum) 'scheme-number)
|
||||
((pair? datum) (car datum))
|
||||
(else (error "Bad tagged datum -- TYPE-TAG" datum))))
|
||||
|
||||
(define (contents datum)
|
||||
(cond ((number? datum) datum)
|
||||
((pair? datum) (cdr datum))
|
||||
(else (error "Bad tagged datum -- CONTENTS" datum))))
|
18
notes.txt
18
notes.txt
|
@ -98,3 +98,21 @@ left and right pointers).
|
|||
Each element of the list is visited once, and for each visit, make-tree is called (which is
|
||||
just a call to list, which is assumed to be O(1)). So list->tree is O(n).
|
||||
|
||||
|
||||
2.77
|
||||
|
||||
It is necessary to call apply-generic for each layer of tagged data. Previously
|
||||
there were only generic functions defined for complex data tagged with 'rect or
|
||||
'polar. Adding the complex package means we call apply-generic once for the complex
|
||||
tag and then again for the 'rect or 'polar tag, using the same real-part, imag-part,
|
||||
magnitude and angle functions. This works because these are defined in terms of
|
||||
apply-generic, which in turn uses the package definition.
|
||||
|
||||
For z as in the example,
|
||||
(magnitude z)
|
||||
(apply-generic 'magnitude z)
|
||||
(magnitude ('rectangular 3 . 4))
|
||||
(apply-generic 'magnitude ('rectangular 3 . 4))
|
||||
(sqrt (+ (square 3) (square 4)))
|
||||
5
|
||||
|
||||
|
|
Loading…
Reference in New Issue