asm*_*asm 6 lisp macros common-lisp quote
我一直在尝试编写一个宏,除其他外,生成一个同名的编译器宏.这是我坚持的最小代码:
(defmacro definline (name lambda-list &body body)
`(define-compiler-macro ,name ,lambda-list
`(let ,,(mapcar (lambda (v) ``(,',v ,,v)) lambda-list)
,,@body)))
Run Code Online (Sandbox Code Playgroud)
我想要的是这样的:
(definline foobar (a b) (print "foobar") (+ a b))
;; Expands to
(define-compiler-macro foobar (a b)
`(let ((a ,a) (b ,b))
(print "foobar") (+ a b)))
Run Code Online (Sandbox Code Playgroud)
但我无法弄清楚如何生成let绑定((a ,a) (b ,b)).我无法解决的问题是如何生成编译器宏表单,以便在扩展中不引用 lambda绑定的内容.我理解如何手动执行此操作,但我不确定如何对任意lambda列表执行此操作.
编辑:
经过一些更多的摆弄,我想出了这个.哪个有效.但是,它很可怕.
(defmacro definline (name lambda-list &body body)
(read-from-string
(format nil "(define-compiler-macro ~S (~{~S~^ ~})
`(let (~{(~S ,~S)~})
~{~S~}))"
name lambda-list (loop for l in lambda-list nconc (list l l)) body)))
Run Code Online (Sandbox Code Playgroud)
如果修改了一些 I/O 相关变量,则使用的方法format根本不起作用。用于format生成符号名称通常非常脆弱且不可移植。但这是一个很难解决的问题,并且使用列表构造而不是单独的反引号来解决它可能更容易。例如,在这种情况下我们可以有:
(defmacro definline (name variables &body body)
(list 'define-compiler-macro name variables
`(list* 'let (list ,@(mapcar (lambda (variable)
`(list (quote ,variable) ,variable))
variables))
',body)))
Run Code Online (Sandbox Code Playgroud)
这可以实现:
CL-USER> (pprint (macroexpand-1 '(definline foobar (a b)
(print "foobar")
(+ a b))))
(DEFINE-COMPILER-MACRO FOOBAR
(A B)
(LIST* 'LET (LIST (LIST 'A A) (LIST 'B B)) '((PRINT "foobar") (+ A B))))
Run Code Online (Sandbox Code Playgroud)
除非我误读了某些内容,否则应该具有与以下相同的结果:
(define-compiler-macro foobar (a b)
`(let ((a ,a) (b ,b))
(print "foobar")
(+ a b)))
Run Code Online (Sandbox Code Playgroud)
我认为单独使用反引号不一定可以生成后一种形式。问题是,由于规范没有准确定义反引号是如何实现的,所以它并不像这样简单
(define-compiler-macro foobar (a b)
(backquote (let ((a (unquote a))
(b (unquote b)))
(print "foobar")
(+ a b)))
Run Code Online (Sandbox Code Playgroud)
如果您的实现确实以这种方式实现它,那么您可以编写一个生成该类型输出的扩展。除非您从实现中得到这样的保证,否则我认为没有办法获得需要注入到更高层的“逗号变量”。很难清楚地说明这一点,但你可以看看这个尝试:
(defmacro definline (name variables &body body)
`(define-compiler-macro ,name ,variables
`(let ,',(mapcar (lambda (variable)
`(,variable (unquote ,variable)))
variables)
,@',body)))
Run Code Online (Sandbox Code Playgroud)
这将产生如下结果:
(DEFINE-COMPILER-MACRO FOOBAR
(A B)
'(LET ((A (UNQUOTE A)) (B (UNQUOTE B)))
(PRINT "foobar")
(+ A B)))
Run Code Online (Sandbox Code Playgroud)
请注意,SBCL 已经足够智能,可以用普通引号替换反引号,因为内部没有任何需要取消引号的内容。生成mapcar拼接形式的 无法生成其中包含逗号的代码,因为它没有指定如何实现这些逗号,并且根据2.4.7 Comma,“如果在除 的正文之外使用,则逗号无效反引号表达式”。我认为这意味着你最好的选择是这样的:
(defmacro definline (name variables &body body)
`(define-compiler-macro ,name ,variables
`(let ,(mapcar 'list
',variables
(list ,@variables))
,@',body)))
Run Code Online (Sandbox Code Playgroud)
在不同的实现下,它的扩展会有所不同,但在 SBCL 中是:
(DEFINE-COMPILER-MACRO FOOBAR (A B)
`(LET (SB-IMPL::BACKQ-COMMA (MAPCAR 'LIST '(A B) (LIST A B)))
(PRINT "foobar")
(+ A B)))
Run Code Online (Sandbox Code Playgroud)
在 CCL 中您将得到:
(DEFINE-COMPILER-MACRO FOOBAR (A B)
(LIST* 'LET
(LIST* (MAPCAR 'LIST '(A B) (LIST A B))
'((PRINT "foobar") (+ A B)))))
Run Code Online (Sandbox Code Playgroud)
在 CLISP 中:
(DEFINE-COMPILER-MACRO FOOBAR (A B)
(CONS 'LET
(CONS (MAPCAR 'LIST '(A B) (LIST A B)) '((PRINT "foobar") (+ A B)))))
Run Code Online (Sandbox Code Playgroud)
| 归档时间: |
|
| 查看次数: |
154 次 |
| 最近记录: |