【发布时间】: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