【问题标题】:LinkageError when using HotswapAgent/DCEVM with Clojure REPL将 HotswapAgent/DCEVM 与 Clojure REPL 一起使用时出现 LinkageError
【发布时间】:2020-05-04 08:38:06
【问题描述】:

我正在尝试在混合 Clojure/Java Leiningen 项目中使用 HotswapAgent/DCEVM,以避免在重新编译 Java 源代码后重新启动 REPL(我已经知道其他方法,例如 JRebel 和 Virgil )。

简而言之,问题是在重新加载我重新编译的 Java 类后,我得到了一个 LinkageError,并且该类似乎没有重新加载。

为了提供更多详细信息,我设置了我的~/.lein/profiles.clj,以便在:repl 配置文件中,我们使用找到here 的当前版本的DCEVM JVM 来运行REPL。这是profiles.clj的相关部分:

{:repl
 {:plugins [[cider/cider-nrepl "0.22.4"]]
  :dependencies [[org.clojure/tools.nrepl "0.2.13"]]

  :java-cmd "/home/jonas/local/dcevm-11.0.6+1/bin/java"

  ... ;; Rest of profiles.clj

为了重现该问题,我使用以下代码设置了一个 minimal mixed Leiningen Clojure and Java project 和一个小型 Java 类 AD

public class AD {
    public double _value;
    public double _deriv;

    public static int EXPONENT = 4;

    public AD(double value, double deriv) {
        _value = value;
        _deriv = deriv;
    }

    public AD mul(AD x) {
        return new AD(_value*x._value, _value*x._deriv + _deriv*x._value);
    }

    public AD raiseToPower() {
        AD result = new AD(1.0, 0.0);
        for (int i = 0; i < EXPONENT; i++) {
            result = result.mul(this);
        }
        return result;
    }

    public String toString() {
        return "AD(value=" + _value + ", deriv=" + _deriv + ")";
    }
}

还有一小段导入此类的 Clojure 代码:

(ns dcevm-complex-demo.ad
  (:import AD))

(defn variable [x]
  (AD. (double x) 1.0))

(defn raise-to-power
  "Evaluates f(x) = x^n and f'(x), n being the AD/EXPONENT static variable"
  [^AD ad-x]
  (.raiseToPower ad-x))

然后我在 Emacs/CIDER 中启动一个 Clojure REPL,加载 dcevm-complex-demo.ad 命名空间并计算表达式 (raise-to-power (variable 3.0)),从而产生结果 AD(value=81.0, deriv=108.0)。然后我修改 Java 源代码,例如,我将行 public static int EXPONENT = 4; 更改为 public static int EXPONENT = 3; 并在终端中使用 lein javac 重新编译。 REPL 中的一条消息告诉我类 AD 已重新加载。但是,当我重新评估表达式 (raise-to-power (variable 3.0)) 时,我希望得到结果 AD(value=27.0, deriv=27.0),但我得到了这个错误:

dcevm-complex-demo.ad/raise-to-power 处的执行错误 (LinkageError) (ad.clj:10)。加载程序约束违规:解析方法'AD时 AD.raiseToPower()' 类加载器 clojure.lang.DynamicClassLoader @7c53a0c2 当前类,dcevm_complex_demo/ad$raise_to_power, 并且该方法的定义类 AD 的类加载器“app”具有 签名中使用的 AD 类型的不同 Class 对象 (dcevm_complex_demo.ad$raise_to_power 位于加载程序的未命名模块中 clojure.lang.DynamicClassLoader @7c53a0c2,父加载器 clojure.lang.DynamicClassLoader @1217a2dd; AD 在未命名的模块中 加载程序“应用程序”)

这是完整的 REPL 交互的样子:

我怎样才能重新加载AD 类?我怀疑我可能必须更改函数 clojure.lang.RT.baseLoader() (clojure/lang/RT.java) 但我不太确定如何去做。

【问题讨论】:

  • 在重新定义之前可以尝试丢弃 DynamicClassLoader.classCache 吗?

标签: java clojure dcevm hotswapagent


【解决方案1】:

在 dcvm8 和 dcvm11 中的 lambda 重新定义存在问题。此问题已在 dcevm v11.0.7+1 中得到解决。根据您的日志,它可能会有所帮助。 https://github.com/TravaOpenJDK/trava-jdk-11-dcevm/releases/tag/dcevm-11.0.7+1

【讨论】:

    猜你喜欢
    • 2010-12-14
    • 1970-01-01
    • 2011-05-04
    • 1970-01-01
    • 1970-01-01
    • 2011-01-11
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多