5

对于 Elisp ,这个问题在某种程度上与this和this 相关。基本上,如何阅读和评估反引号?正在发生什么过程?标准对此有什么说明吗?

这是我所期望的,但它不会发生:符号`是一个阅读器宏,被翻译成某种(BACKQUOTE ...)宏/特殊形式(类似于'被翻译成(QUOTE ...))。这不会发生,事实上,Common Lisp 甚至没有BACKQUOTE宏。

发生了什么(SBCL):

CL-USER> (defparameter *q* (read-from-string "`(a b ,c)"))
*Q*
CL-USER> *q*
`(A B ,C)
CL-USER> (car *q*)
SB-INT:QUASIQUOTE
CL-USER> (cdr *q*)
((A B ,C))

与预期不同的东西,但还可以。现在,,C它本身就是一个有趣的野兽:

CL-USER> (type-of (third (cadr *q*)))
SB-IMPL::COMMA

如果没有逗号符号,评估读取表达式就可以了:

CL-USER> (eval (read-from-string "`(a b c)"))
(A B C)

但是,如果我想评估原始表达式,即使使用本地绑定 for C,也会出现问题:

(let ((c 10)) (eval (read-from-string "`(a b ,c)")))
; in: LET ((C 10))
;     (LET ((C 10))
;       (EVAL (READ-FROM-STRING "`(a b ,c)")))
; 
; caught STYLE-WARNING:
;   The variable C is defined but never used.
; 
; compilation unit finished
;   caught 1 STYLE-WARNING condition
; Evaluation aborted on #<UNBOUND-VARIABLE C {1007A3B2F3}>.

这意味着EVAL没有拾取C绑定的环境。

PS。有趣的是,在 Elisp 中这是可行的。

4

1 回答 1

13

反引号

反引号是 Common Lisp中的标准宏字符

在 Common Lisp 中,反引号表达式的表示是未定义的。实现实际上使用不同的表示。您看到的 SBCL 是特定于实现的。

评估

eval的问题与阅读器或反引号表达式完全无关:

? (let ((c 10))
    (eval '(list 'a 'b c)))

Error: The variable C is unbound.

在 Common LispEVAL中,使用动态环境和空词法环境来评估表单。c上面绑定到 的词法环境10没有被使用。

但是动态绑定是。我们需要将变量声明为 special

? (let ((c 10))
    (declare (special c))
    (eval '(list 'a 'b c)))
(A B 10)

因此,这也有效:

? (let ((c 10))
    (declare (special c))
    (eval (read-from-string "`(a b ,c)")))
(A B 10)

Emacs Lisp 默认有/有动态绑定(尽管 GNU Emacs 现在也支持词法绑定)。Common Lisp 默认有词法绑定。

于 2016-03-05T11:07:17.660 回答