Lisp:在宏中扩展属性名称

问题描述 投票:1回答:1

考虑这个属性列表:

(defvar *some-variable* (list :foo "fooval" :bar "barval"))

这个简单的电话:

(getf *some-variable* :foo)

按预期收益"fooval"。我定义了一个宏应该做同样的事情,除了我可以传递任何属性的名称来检索:

(defmacro my-macro (property-name)
    `(getf *some-variable* :,property-name))

不幸的是,这样称呼它:

(my-macro 'foo)

结果在FOO。为什么?

macros lisp common-lisp clisp gnu-common-lisp
1个回答
5
投票

你为什么不亲自检查一下:

(macroexpand-1 '(my-macro 'foo))
; ==> (getf *some-variable* :|| 'foo) ;
T

documentation for getf说,如果你给它一个第四个参数,它是找不到密钥时的值。由于:||(关键字包中的空符号)不存在,因此返回提供的默认foo

所以这是一个能够做你想要的功能:

(defun get-field (name)
 (getf *some-variable* 
       (intern (symbol-name name) "KEYWORD")))

(defparameter *test* 'foo)
(get-field *test*)
; ==> "fooval"

使其成为宏的唯一原因是使其成为语法,语法和函数之间的主要区别在于不评估参数。

(defmacro get-mfield (name)
  `(get-field ',name))

(get-mfield foo)
; ==> "fooval"

(get-mfield *test*)
; ==> nil

你得到文字裸露,但你失去了*test*被视为变量而不是关键:*test*的特征

© www.soinside.com 2019 - 2024. All rights reserved.