似乎我对宏的理解是不完整的。 如何将参数传递给宏,并从函数中调用它们。 代码下面看似简单的代码不起作用。
(defmacro bar [arg]
(println (symbol? arg)))
(defn foo [arg]
(bar arg))
(foo 'baz) => nil
答案 0 :(得分:6)
定义foo时正在评估宏。如果在repl中定义foo函数,则会注意到Clojure打印为true。
user=> (defmacro bar [arg]
(println (symbol? arg)))
#'user/bar
user=> (defn foo [arg]
(bar arg))
true ; <= This is the print from your macro
; (it prints true, because arg is a symbol)
#'user/foo
你需要引用宏的主体,否则它将被评估,因为它的返回值是nil,foo将只返回nil。
(defmacro bar [arg]
`(println (symbol? ~arg)))
答案 1 :(得分:4)
宏应该生成代码。它将源表达式作为参数并创建新的源。
让我们看看:
user=> (defmacro bar [arg]
(println (symbol? arg)))
#'user/bar
user=> (bar 1)
false
nil
user=> (bar 'a)
false
nil
user=> (bar a)
true
nil
这是扩展:
user=> (macroexpand-1 '(bar a))
true
它不会生成任何有用的代码。
答案 2 :(得分:3)
要使宏“工作”,它需要返回代码,通常作为列表表达式。因此,出于您的意图,您需要提供:
(defmacro bar (arg)
`(println (symbol? ,arg)))
(这使用quasiquote作为构造列表的方便表示法。)在使用bar
时,通过运行defmacro代码扩展使用以生成该列表。然后编译该列表。例如:
(progn (bar 10))
扩展为
(progn (println (symbol? 10))
编译后再运行代码并打印'nil'。您会注意到(bar a)
会产生错误“a is not bound
”,因为扩展为(println (symbol a))
,评估a
,可能没有值。 (bar 'a)
会返回T
。
使用这个正确的bar
宏,您的函数foo
在编译时将扩展为:
(defun foo (x)
(println (symbol? x))
正确计算(foo 'a)
和(foo 10)
。