0

我可以使用memfn创建一个调用 java 函数的 clojure 函数。

(macroexpand '(memfn startsWith prefix))
=> (fn* ([target2780 prefix] (. target2780 (startsWith prefix))))
((memfn startsWith prefix) "abc" "a")
=> true

memfn要求函数名是符号。我想知道是否可以编写一个宏来调用名称以字符串形式提供的任意方法。也就是说,我希望能够调用以下内容:

(def fn-name "startsWith")
=> #'user/fn-name
(macroexpand '(memfn' fn-name "prefix"))
=> (fn* ([target2780 prefix] (. target2780 (startsWith prefix))))
((memfn fn-name "prefix") "abc" "a")
=> true

我能想到的唯一方法是使用read-string.

(defmacro memfn' [fn-name arg-name]
  `(memfn ~(read-string fn-name) ~arg-name))

编辑:使用 read-string 和 eval 的版本实际上按我想要的方式工作。

(defn memfn' [fn-name arg-name]
  (eval (read-string (str "(memfn " fn-name " " arg-name ")"))))

我是否缺少一个基本的宏构建工具来获取符号引用的字符串并将其转换为文字而不可能执行代码read-string

感谢您的任何想法!

4

2 回答 2

0

无论有没有读字符串,都没有办法做到这一点。您提出的解决方案不起作用。您真正想要区分的不是字符串和符号,而是运行时数据和编译时文字。宏不会评估它们收到的参数,因此即使fn-name是 var 的名称,其值为"startsWith", memfn(或您的memfn'宏)也只会看到fn-name.

于 2018-11-29T03:26:05.177 回答
0

如果您只对调用 java 方法感兴趣,那么您可以依赖 java.lang.reflect.Method 及其调用方法。

像这样的东西应该适用于无参数方法,并且不需要宏。

(defn memfn' [m]
  (fn [o] (.invoke (.getMethod (-> o .getClass) m nil) o nil)))

((memfn' "length") "clojure")
;=>7
于 2018-11-30T21:05:52.323 回答