Java互操作的Clojure优化

时间:2014-02-27 18:41:10

标签: clojure clojure-java-interop

当使用现有的java类时,如果我做错了某些事情,我经常会收到反射警告,例如

  

IllegalArgumentException找不到匹配的字段:获取类   java.lang.String clojure.lang.Reflector.getInstanceField   (Reflector.java:271)

在每次调用给定方法时,clojure是否在运行时进行反射?或者这是以任何方式缓存的?将任何类型的java-interop移动到相关的java类中是否会有速度优势?

2 个答案:

答案 0 :(得分:5)

Clojure只有在无法根据周围的上下文推断出调用的确切方法时才会在运行时进行反射,否则会发出直接调用该方法的代码。如果需要,您可以使用类型提示为编译器提供此上下文。例如:

user=> (set! *warn-on-reflection* true)

user=> (defn str-len [x] (.length x))
Reflection warning, NO_SOURCE_PATH:1:19 - reference to field length can't be resolved.

user=> (defn str-len-2 [^String x] (.length x))

user=> (str-len "abc") ; -> 3
user=> (str-len-2 "abc") ; -> 3

user=> (time (dotimes [_ 100000] (str-len "abc")))
"Elapsed time: 1581.163611 msecs"
user=> (time (dotimes [_ 100000] (str-len-2 "abc")))
"Elapsed time: 36.838201 msecs"

第一个函数每次调用时都会使用反射;第二个与本机Java代码具有相似的性能。

答案 1 :(得分:4)

这不是反射警告,只是表明它正在使用反射。

您可以使用type hints来消除反射。上面链接中描述的*warn-on-reflection*标志(默认为false),可选择启用反射警告。

我发现使用Leiningen的lein check实用程序很方便,该实用程序会尝试编译项目中的每个Clojure文件,并启用反射警告。这将报告代码或从库中加载的任何代码中的反射问题。