不确定你使用的是哪个版本的 Clojure 或编辑器,但 Clojure 1.10.3 在标准 repl 中给出更多有用的输出
user=> (macroexpand '(m))
Syntax error macroexpanding clojure.core/fn at (REPL:1:1).
(user/x) - failed: Extra input at: [:fn-tail :arity-1 :params] spec: :clojure.core.specs.alpha/param-list
user/x - failed: vector? at: [:fn-tail :arity-n :params] spec: :clojure.core.specs.alpha/param-list
如你所见,它并不是运行你的代码,而是在遇到问题时(递归地)展开宏。第一次展开将变成
(clojure.core/fn [user/x])
clojure.core/fn
本身就是一个宏,它在展开时被检查是否符合 fn
规范(这是默认开启的宏规范)。规范发现 user/x
并将其视为 ":params" 中的“多余输入”,因为它们不符合任何预期的参数形式(通常是不带限定符的符号,但也包括所有递归解构形式)。实际上,这里的问题是 user/x
而不是 x
。
你可以通过以下方式修复这个问题
(defmacro m []
`(fn [~'x]))