我正在尝试使用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 交互:
我怎样才能重新加载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/