There’s another difficulty with using substitution, which is the number of times we traverse the source program. It would be nice to have to traverse only those parts of the program that are actually evaluated, and then, only when necessary.
6.1 Introducing the Environment
三个直觉
1, have the interpreter “look up” an identifier in some sort of directory 2,defer the substitution. 3, The directory records the intent to substitute,without actually rewriting the program source.
Each name association in the environment is called a binding.
Observe carefully that what we are changing is the implementation strategy for the programming language, not the language itself.
Let’s first define our environment data structure. An environment is a list of pairs of names associated with...what?
A:Remember that our environment was created to defer substitutions. Therefore, the answer lies in substitution.
答案是值。
We discussed earlier (section 5.5) that we want substitution to map names to answers, corresponding to an eager function application strategy.
#lang plai-typed
(define-type ExprS
[numS (n : number)]
[plusS (l : ExprS) (r : ExprS)]
[bminusS (l : ExprS) (r : ExprS)]
[uminusS (e : ExprS)]
[multS (l : ExprS) (r : ExprS)]
[idS (s : symbol)]
[appS (fun : symbol) (arg : ExprS)])
(define-type ExprC
[numC (n : number)]
[idC (s : symbol)]
[appC (fun : symbol) (arg : ExprC)]
[plusC (l : ExprC) (r : ExprC)]
[multC (l : ExprC) (r : ExprC)])
(define-type FunDefC
[fdC (name : symbol) (arg : symbol) (body : ExprC)])
(define (parse [s : s-expression]) : ExprS
(cond
[(s-exp-number? s) (numS (s-exp->number s))]
[(s-exp-symbol? s) (idS (s-exp->symbol s))]
[(s-exp-list? s)
(let ([sl (s-exp->list s)])
(case (s-exp->symbol (first sl))
[(+) (plusS (parse (second sl)) (parse (third sl)))]
[(*) (multS (parse (second sl)) (parse (third sl)))]
[(-) (bminusS (parse (second sl)) (parse (third sl)))]
[(u-) (uminusS (parse (second sl)))]
[else (appS (s-exp->symbol (first sl))
(parse (second sl)))]))]
[else (error 'parse "invalid input")]))
(define (parsedef [s : s-expression]) : FunDefC
(cond
[(s-exp-list? s)
(let ([sl (s-exp->list s)])
(case (s-exp->symbol (first sl))
[(define) (fdC (s-exp->symbol (first (s-exp->list (second sl))))
(s-exp->symbol (second (s-exp->list (second sl))))
(desugar (parse (third sl))))]
[else (error 'parsedef "invalid list")]))]
[else (error 'parsedef "invalid input")]))
(define (desugar [as : ExprS]) : ExprC
(type-case ExprS as
[numS (n) (numC n)]
[plusS (l r) (plusC (desugar l)
(desugar r))]
[multS (l r) (multC (desugar l)
(desugar r))]
[bminusS (l r) (plusC (desugar l)
(multC (numC -1) (desugar r)))]
[uminusS (e) (multC (numC -1)
(desugar e))]
[idS (s) (idC s)]
[appS (fun arg) (appC fun (desugar arg))]))
(define (subst [what : ExprC] [for : symbol] [in : ExprC]) : ExprC
(type-case ExprC in
[numC (n) in]
[idC (s) (cond
[(symbol=? s for) what]
[else in])]
[appC (f a) (appC f (subst what for a))]
[plusC (l r) (plusC (subst what for l)
(subst what for r))]
[multC (l r) (multC (subst what for l)
(subst what for r))]))
(define (get-fundef [n : symbol] [fds : (listof FunDefC)]) : FunDefC
(cond
[(empty? fds) (error 'get-fundef "reference to undefined function")]
[(cons? fds) (cond
[(equal? n (fdC-name (first fds))) (first fds)]
[else (get-fundef n (rest fds))])]))
(define (lookup [n : symbol] [env : Env]) : number
(cond
[(empty? env) (error 'lookup "Symbol not found in env")]
[(cons? env) (cond
[(equal? n (bind-name (first env))) (bind-val (first env))]
[else (lookup n (rest env))])]))
(define-type Binding
[bind (name : symbol) (val : number)])
(define-type-alias Env (listof Binding))
(define mt-env empty)
(define extend-env cons)
(define (interp [expr : ExprC] [env : Env] [fds : (listof FunDefC)]) : number
(type-case ExprC expr
[numC (n) n]
[idC (n) (lookup n env)]
[appC (f a) (local ([define fd (get-fundef f fds)])
(interp (fdC-body fd)
(extend-env (bind (fdC-arg fd)
(interp a env fds))
mt-env)
fds))]
[plusC (l r) (+ (interp l env fds) (interp r env fds))]
[multC (l r) (* (interp l env fds) (interp r env fds))]))
(define (driver e fdlist)
(interp (desugar (parse e)) mt-env (map parsedef fdlist)))
(driver `(f 1) (list `(define (f x) (+ x x))))
(driver `(g 1) (list `(define (f x) (+ x x))
`(define (g x) (f x))))