为什么读取器宏扩展不会传播到运行时(读取)?

时间:2013-09-02 13:53:48

标签: lisp common-lisp reader-macro

为什么以下不起作用?

;;;; foo.lisp
(in-package :cl-user)

(eval-when (:compile-toplevel :load-toplevel :execute)
  (require :cl-interpol))

(cl-interpol:enable-interpol-syntax)

(defun read-and-eval (s)
  (eval (read-from-string s)))

(cl-interpol:disable-interpol-syntax)

然后:

LISP> (load (compile-file "foo.lisp"))
=> T

LISP> (read-and-eval
        "(let ((a \"foo\")) (princ #?\"${a}\"))")
=> no dispatch function defined for #\?

2 个答案:

答案 0 :(得分:10)

因为只有一个读者,具有全局状态。您可以有效地打开和关闭宏。在这种情况下,读取器宏仅在编译时读取read-and-eval函数的持续时间内启用。

在这种情况下,您需要在read-and-eval函数内设置宏,以确保读者在需要时处于正确的状态。

答案 1 :(得分:8)

CL:根据在调用READ运行时绑定到CL:* READTABLE *的可读表进行调度。在引擎盖下,ENABLE-INTERPOL-SYNTAX正在创建一个新的可读表,设置CL:* READTABLE *来保存它,并存储旧的CL:* READTABLE *。 DISABLE-INTERPOL-SYNTAX打开先前的可读表并设置CL:* READTABLE *再次保持它。最小化您的原始设置,您可以通过以下方式安排您想要的行为:

(in-package :cl-user)

(eval-when (:compile-toplevel :load-toplevel :execute)
  (require :cl-interpol))

(cl-interpol:enable-interpol-syntax)

(defvar *interpol-reader* *readtable*)

(cl-interpol:disable-interpol-syntax)

(defun read-and-eval (s)
  (let ((*readtable* *interpol-reader*))
    (eval (read-from-string s))))

禁用语法的调用可以放在defvar和read-and-eval之后的任何位置,但是如果你想在文件中直接输入interpol语法,则必须在enable和disable之间放置语法调用。对于后一个目的,国际刑警组织的呼叫扩展到EVAL-WHENs是很重要的,原因是你要求REQUIRE在EVAL-WHEN范围内;也就是说,当后者的形式是READ时,效果需要已经发生。

CL-INTERPOL的界面抽象了正在发生的事情,因此我将向您展示如何手动创建和更改可读表:

;; Create a fresh readtable with standard syntax
(defvar *not-readtable* (copy-readtable nil))

;; A simple reader function
(defun not-reader (stream char &optional count)
  "Like ' but for (not ...) instead of (quote ...)"
  (declare (ignore count char))
  `(not ,(read stream t nil t)))

;; Mutate that readtable so that the dispatch character you want
;; calls the function you want
(set-macro-character #\! 'not-reader nil *not-readtable*)

;; Try it out
(let ((*readtable* *not-readtable*))
  (read-from-string "(if !foo bar baz)"))

=>
(IF (NOT FOO)
    BAR
    BAZ)