5

在使用现有的 java 类时,如果我做错了什么,我经常会收到反射警告,例如

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

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

4

2 回答 2

6

只有当 Clojure 无法根据周围的上下文推断要调用的确切方法时,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 代码相似的性能。

于 2014-02-27T18:56:07.080 回答
4

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

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

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

于 2014-02-27T18:52:30.857 回答