SICP問題4.6

2010/03/17 evalでlookup-variable-valueのトコが間違ってたので修正

let式が

(let ((<var1> <exp1>) ... (<varn> <expn>))
  <body>)

lambda式

((lambda (<var1> ... <varn>)
    <body>)
 <exp1>
 ...
 <expn>)

のsyntax sugarであることから、変換let->combinationを実装し、let式が使えるようevalに適当な節を追加する。
let関連の手続き

(define (let? exp)
  (tagged-list? exp 'let))
(define (let-parameters exp)
  (cadr exp))
(define (let-body exp)
  (cddr exp))
(define (let-variables exp)
  (map car (let-parameters exp)))
(define (let-expressions exp)
  (map cadr (let-parameters exp)))
(define (let->combination exp)
  (if (null? (let-parameters exp))
      '()
    (cons
     (make-lambda (let-variables exp) (let-body exp))
     (let-expressions exp))))

(データ主導型でない)evalへの組み込み

(define (eval exp env)
  (cond ((self-evaluating? exp) exp)
        ((variable? exp) (lookup-variable-value exp env))
        ((quoted? exp) (text-of-quotation exp))
        ((assignment? exp) (eval-assignment exp env))
        ((definition? exp) (eval-definition exp env))
        ((if? exp) (eval-if exp env))
        ((lambda? exp)
         (make-procedure (lambda-parameters exp)
                         (lambda-body exp)
                         env))
        ((let? exp) (eval (let->combination exp) env)) ; 追加
        ((begin? exp)
         (eval-sequence (begin-actions exp) env))
        ((cond? exp) (eval (cond->if exp) env))
        ((and? exp) (eval-and exp env))
        ((or? exp) (eval-or exp env))
        ((application? exp)
         (apply (eval (operator exp) env)
                (list-of-values (operands exp) env)))
        (else
         (error "Unknown expression type -- EVAL" exp))))