在使用现有的 java 类时,如果我做错了什么,我经常会收到反射警告,例如
IllegalArgumentException 未找到匹配字段:获取类 java.lang.String clojure.lang.Reflector.getInstanceField (Reflector.java:271)
clojure 是否在运行时为给定方法的每次调用进行反射?还是以任何方式缓存?将任何涉及的 java-interop 移动到相关的 java 类中会有速度优势吗?
在使用现有的 java 类时,如果我做错了什么,我经常会收到反射警告,例如
IllegalArgumentException 未找到匹配字段:获取类 java.lang.String clojure.lang.Reflector.getInstanceField (Reflector.java:271)
clojure 是否在运行时为给定方法的每次调用进行反射?还是以任何方式缓存?将任何涉及的 java-interop 移动到相关的 java 类中会有速度优势吗?
只有当 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 代码相似的性能。
这不是反射警告,它只是表明它正在使用反射。
您可以使用类型提示来消除反射。*warn-on-reflection*
上述链接中描述的标志(默认为 false),可选择启用反射警告。
我发现使用 Leiningen 的实用程序很方便lein check
,它会尝试编译项目中的每个 Clojure 文件,并打开反射警告。这将报告您的代码或从库加载的任何代码中的反射问题。