Review section
14.4 in the book, including the problems. Now, consider these data
definitions:
; a Value is one of: Number String
;
; a Prim is one of:
; '+ '- '* '/ 'sqrt
; 'string-length 'string-append 'number->string
;
; a PExp (primitive expression) is one of: Value PApp
;
; a PApp (primitive application) is a (cons Prim [Listof PExp])
In all of the following, use loop functions such as map
and member?
where possible, and consider using
apply
where appropriate as well. You may find it easier to
first implement the functions without these "power tools" and then look for
ways to simplify the code by using them.
Part 1: fill in the ...
in the following functions,
and make sure all the tests pass:
; prim?: Any -> Boolean
; is the argument a primitive?
(define (prim? v) ...)
(check-expect (prim? '+) true)
(check-expect (prim? 'number->string) true)
(check-expect (prim? 'foo) false)
; apply-prim: Prim [Listof Value] -> Value
; apply a primitive to a list of parameter values
; the primitives '+ '- '* '/ and 'string-append accept two or more inputs each
; the remaining primitives accept one input only
(define (apply-prim fn vals) ...)
(check-expect (apply-prim '+ '(1 2 3)) 6)
(check-expect (apply-prim '- '(1 2 3)) -4)
(check-expect (apply-prim '* '(1 2 3)) 6)
(check-expect (apply-prim '/ '(1 2)) (/ 1 2))
(check-expect (apply-prim 'sqrt '(4)) 2)
(check-expect (apply-prim 'string-length '("foo")) 3)
(check-expect (apply-prim 'string-append '("a" "b")) "ab")
(check-expect (apply-prim 'number->string '(23)) "23")
Part 2: fill in the ...
in the following function, and
make sure all the tests pass:
; prim-eval: PExp -> Value
; evaluate a primitive expression
(define (prim-eval s) ...)
(check-expect (prim-eval 2) 2)
(check-expect (prim-eval "a") "a")
(check-expect (prim-eval '(+ 3 (* 2 3))) 9)
(check-expect (prim-eval '(string-append "a" (number->string 2))) "a2")
(check-expect (prim-eval '(* (string-length "bob") 2)) 6)
Congratulations, you have now made a Scheme expression interpreter that
can handle numeric and string operations, including nested subexpressions!
However, it does not know about user-defined functions...
Review section
17.7 in the book, including the problems. Then consider these data
definitions, in addition to those from Problem A1:
; a Name is a Symbol
;
; a Func is one of: Prim Name
;
; a SExp (simple scheme expression) is one of: Value Name App
;
; an App (application) is a (cons Func [Listof SExp])
;
; a Def (definition) is a (make-def Name [Listof Name] SExp)
(define-struct def (name args body))
And this "starter" list of definitions:
(define the-defs (list (make-def 'pi '() 3.14)
(make-def 'length '(x y) '(sqrt (+ (* x x) (* y y))))
(make-def 'circumference '(r) '(* 2 pi r))
(make-def 'prepend-bob '(s) '(string-append "bob" s))))
In all of the following, use loop functions such as map
where possible, and consider using lambda
where appropriate as
well. And, of course, use your solutions from problem A1 as needed.
Part 1: fill in the ...
in the following function, and
make sure all the tests pass:
; lookup: Name Number [Listof Def] -> Def
; find the definition with the given name and number of args in defs
; or (error name " not defined")
(define (lookup name nargs defs) ...)
(check-expect (def-body (lookup 'pi 0 the-defs)) 3.14)
(check-expect (def? (lookup 'length 2 the-defs)) true)
(check-error (lookup 'length 3 the-defs))
(check-error (lookup 'sqr 1 the-defs))
Part 2: fill in the ...
in the following function, and
make sure all the tests pass:
; subst: [Listof Value] [Listof Name] SExp -> SExp
; substitute the corresponding value for each listed variable name in s
; leave all other names in s unmodified
(define (subst vals vars s) ...)
(define s1 '(foo a 29 (bar "z" b)))
(check-expect (subst '(3 "x") '(a b) s1) '(foo 3 29 (bar "z" "x")))
Part 3: fill in the ...
in the following functions, and
make sure all the tests pass:
; my-apply: Func [Listof Value] [Listof Def] -> Value
; apply either a primitive or a defined function to a list of parameter values
(define (my-apply fn vals defs) ...)
; my-eval: SExp [Listof Def] -> Value
; evaluate a scheme expression given a list of definitions
(define (my-eval s defs) ...)
(check-expect (my-apply '+ '(1 2 3) the-defs) 6)
(check-expect (my-apply 'length '(3 4) the-defs) 5)
(check-error (my-apply 'sqr '(2) the-defs))
(check-error (my-apply 'circumference '(1 2) the-defs))
(check-expect (my-eval 2 the-defs) 2)
(check-expect (my-eval "b" the-defs) "b")
(check-expect (my-eval '(+ 3 4) the-defs) 7)
(check-expect (my-eval 'pi the-defs) 3.14)
(check-expect (my-eval '(* 2 pi) the-defs) 6.28)
(check-expect (my-eval '(prepend-bob (number->string 18)) the-defs) "bob18")
(check-expect (my-eval '(circumference 2) the-defs) (* 4 3.14))
(check-expect (my-eval '(circumference (/ 10 (length 3 4))) the-defs) (* 4 3.14))
(check-expect (my-eval '(* (string-length (string-append "a" "b")) 4) the-defs) 8)
(check-error (my-eval '(sqr 4) the-defs))
(check-expect (my-eval '(sqr 4) (cons (make-def 'sqr '(x) '(* x x)) the-defs)) 16)
Hint: my-apply
and my-eval
should be mutually
recursive.
Double congratulations! You have now extended your Scheme interpreter to
handle user-defined names for both values (when the number of arguments in
the definition is 0) and functions with one or more named parameters
(i.e. variables).
Challenge: This assignment is not easy. That said, it is possible to
solve it in less than 45 lines, not counting comments and tests, without
going over 80 characters on any line, and without sacrificing readability.
That's about 35 additional lines of code to finish out the code we
provided above. It's pithy code, but not a lot of it. (The graders will
consider clarity and conciseness in evaluating your code, but you do not have
to strive to make it as short as possible. Just try to make a good clear
design.)