【问题标题】:Clojure optimization of java interopjava互操作的Clojure优化
【发布时间】:2014-03-31 09:53:57
【问题描述】:
在使用现有的 java 类时,如果我做错了某些事情,我经常会收到反射警告,例如
IllegalArgumentException 未找到匹配字段:获取类
java.lang.String clojure.lang.Reflector.getInstanceField
(Reflector.java:271)
clojure 是否在运行时对给定方法的每次调用进行反射?还是以任何方式缓存?将涉及的任何类型的 java-interop 移动到相关的 java 类中会有速度优势吗?
【问题讨论】:
标签:
clojure
clojure-java-interop
【解决方案1】:
这不是反射警告,它只是表明它正在使用反射。
您可以使用type hints 来消除反射。上面链接中描述的*warn-on-reflection* 标志(默认为 false),可选择启用反射警告。
我发现使用 Leiningen 的 lein check 实用程序很方便,它会尝试编译项目中的每个 Clojure 文件,并打开反射警告。这将报告您的代码或从库加载的任何代码中的反射问题。
【解决方案2】:
只有在无法根据周围上下文推断要调用的确切方法时,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 代码相似的性能。