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))))