2

我在clojure中创建了一个宏

(ns macro-question.map)

(defmacro lookup [key] (list get (apply hash-map (range 1 5)) key))

在 clojure repl 中它按预期工作

$ clj
Clojure 1.9.0
user=> (require 'macro-question.map)
nil
user=> (macro-question.map/lookup 1)
2

所以我创建了一个这样的 clojurescript 模块来尝试使用它:

(ns macro-question.core (:require-macros [macro-question.map]))

(defn lookup1 [] (macro-question.map/lookup 1))

当我在repl中尝试时,我得到了这个

$ clj -m cljs.main  --repl-env node
ClojureScript 1.10.520
cljs.user=> (require 'macro-question.core)
Execution error (ExceptionInfo) at cljs.compiler/fn (compiler.cljc:304).
failed compiling constant: clojure.core$get@3df1a1ac; clojure.core$get is not a valid ClojureScript constant.

同时回到clojure,有一个线索为什么这可能是

user=> (macroexpand '(macro-question.map/lookup 1))
(#object[clojure.core$get 0x101639ae "clojure.core$get@101639ae"] {1 2, 3 4} 1)

我可以创建以'(而不是(list. 但是,我希望在构建时扩展地图。

到底是怎么回事?我该怎么做才能获得以下内容:

user=> (macroexpand '(macro-question.map/lookup 1))
(get {1 2, 3 4} 1)

或者我应该怎么做才能在 clojurescript 中使用这个宏?

4

2 回答 2

1

(list get (apply hash-map (range 1 5)) key)

创建一个列表,其中第一个位置是 varget引用的函数对象。您真正想要返回的是一个列表,其中第一个位置的完全限定符号get将您的宏定义更改为

(defmacro lookup [key] 
  `(get ~(apply hash-map (range 1 5)) ~key))
(macroexpand '(lookup 1))
=> (clojure.core/get {1 2, 3 4} 1)

读者参考指南在这里很有帮助https://clojure.org/reference/reader

于 2019-11-21T19:44:36.887 回答
1

你可以放'get!!

(ns macro-question.map)

(defmacro lookup [key] (list 'get (apply hash-map (range 1 5)) key))

https://stackoverflow.com/posts/58985564是关于它为什么会发生的更好答案,也是一种更好的方法。这只是显示了一个简单的解决方案,它只解决了眼前的问题,我在我之后就意识到了问了这个问题)

于 2019-11-22T00:01:38.690 回答