Представление анонимного макроса по соглашению представляет собой список вида (macro lambda ...)
.Попробуйте проверить их в своем любимом интерпретаторе Lisp (проверено в Emacs):
> (defmacro triple (x) `(+, x, x, x))
тройной
> (символ-функция 'тройной)
(макрос лямбда (x) (\ `(+ (\,x) (\, x) (\, x))))
Хотя в Emacs все работает не так, остается только дать адекватную семантику такойформа.То есть, когда eval.
видит ((macro lambda (x) EXPR) FORM)
, он должен
- Заменить каждый случай
x
в FORM
на EXPR
без , сначала оценивая EXPR
(в отличие от того, что происходит при вызове функции); eval.
результат выше.
Этого можно добиться, добавив предложение к самому внешнему cond
вeval.
, что касается дела ((macro lambda ...) ...)
.Вот грубый прототип:
((eq (caar e) 'macro)
(cond
((eq (cadar e) 'lambda)
(eval. (eval. (car (cdddar e))
(cons (list. (car (caddar e)) (cadr e)) a))
a))))
Этот код работает только для макросов с одним аргументом.Исправление включает в себя написание вспомогательной функции substlis.
, которая работает как evlis.
, но без зацикливания на eval.
;это оставлено в качестве упражнения для читателя: -)
Для проверки определите cadr.
как макрос следующим образом:
(defmacro cadr. (x)
(list. 'car (list. 'cdr x)))
После этого вы получите
> (символьная функция 'cadr.)
(макрос lambda (x) (list. (Quote car) (list. (Quote cdr) x)))
Вы можете создать форму, которая применяет это (macro lambda ...)
к выражению, и оценить эту конструкцию в контексте, который содержит определение для list.
(поскольку интерпретатор eval.
не считает его примитивным).).Например,
(let ((e '((macro lambda (x) (list (quote car) (list (quote cdr) x)))
(cons (quote x) (cons (quote y) nil))))
(bindings `((list ,(symbol-function 'list.)))))
(eval. e bindings))
y
Тада!