【问题标题】: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 代码相似的性能。

      【讨论】:

      • 谢谢,这很有帮助。
      猜你喜欢
      • 2015-06-12
      • 1970-01-01
      • 1970-01-01
      • 2019-07-18
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多