java - 将 HotswapAgent/DCEVM 与 Clojure REPL 结合使用时出现 LinkageError

标签 java clojure dcevm hotswapagent

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

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

为了提供更多详细信息,我设置了 ~/.lein/profiles.clj,以便在 :repl 配置文件中,我们使用当前版本的 DCEVM找到 JVM here运行 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; 并使用 重新编译在终端中输入 javac 。 REPL 中的一条消息告诉我类 AD 已重新加载。但是,当我重新评估表达式 (raise-to-power (variable 3.0)) 时,我期望结果 AD(value=27.0, deriv=27.0),但我得到了这个错误:

Execution error (LinkageError) at dcevm-complex-demo.ad/raise-to-power (ad.clj:10). loader constraint violation: when resolving method 'AD AD.raiseToPower()' the class loader clojure.lang.DynamicClassLoader @7c53a0c2 of the current class, dcevm_complex_demo/ad$raise_to_power, and the class loader 'app' for the method's defining class, AD, have different Class objects for the type AD used in the signature (dcevm_complex_demo.ad$raise_to_power is in unnamed module of loader clojure.lang.DynamicClassLoader @7c53a0c2, parent loader clojure.lang.DynamicClassLoader @1217a2dd; AD is in unnamed module of loader 'app')

这就是完整的 REPL 交互:

REPL interaction

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

最佳答案

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

关于java - 将 HotswapAgent/DCEVM 与 Clojure REPL 结合使用时出现 LinkageError,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/61588170/

相关文章:

java - 如何在Sphinx4中评估样本

java - 处理多线程/传出 http 请求

java - 在java中读取文件时遇到问题

java - Joda - 从 PeriodFormat.getDefault().print() 中删除秒、毫秒?

clojure - Clojure Core 或 Contrib 中是否有等效的 Zip 函数?

java - 如何在没有 IDE 的情况下重新加载更改/HotSwap 重新加载类

java - HotswapAgent org.hotswap.agent.command.ReflectionCommand异常的原因?

asynchronous - Clojurescript:使用核心/异步 channel 以 block 的形式处理请求

Clojure STM 歧义因子

virtual-machine - 类似于具有 Java 7 支持的动态代码演化的工具