clojure:Namespace中的ExceptionInInitializerError。<init>从非默认类路径加载</init>

时间:2012-03-21 22:48:44

标签: clojure classloader

在尝试从非默认类路径加载AOT编译的类时,我收到以下异常:

Traceback (innermost last):
  File "test.jy", line 10, in ?
        at clojure.lang.Namespace.<init>(Namespace.java:34)
        at clojure.lang.Namespace.findOrCreate(Namespace.java:176)
        at clojure.lang.Var.internPrivate(Var.java:149)
        at aot_demo.JavaClass.<clinit>(Unknown Source)
        at java.lang.Class.forName0(Native Method)
        at java.lang.Class.forName(Class.java:247)
        at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
        at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
        at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
        at java.lang.reflect.Method.invoke(Method.java:597)
java.lang.ExceptionInInitializerError: java.lang.ExceptionInInitializerError

我可以使用以下简单的project.clj重现这一点:

(defproject aot-demo "0.1.0-SNAPSHOT"
  :dependencies [[org.clojure/clojure "1.3.0"]]
  :aot [aot-demo.core])

...和src/aot_demo/core.clj定义如下:

(ns aot-demo.core
  (:gen-class
    :name aot_demo.JavaClass
    :methods [#^{:static true} [lower [java.lang.String] java.lang.String]]))

(defn -lower [str] (.toLowerCase str))

以下Jython脚本足以触发错误:

#!/usr/bin/jython
import java.lang.Class
import java.net.URLClassLoader
import java.net.URL
import os

customLoader = java.net.URLClassLoader(
    [java.net.URL('file://%s/target/aot-demo-0.1.0-SNAPSHOT-standalone.jar'
                  % (os.getcwd()))])

java.lang.Class.forName('aot_demo.JavaClass', True, customLoader)

但是,如果测试脚本是以CLASSPATH变量中已经存在的uberjar启动的,则会发生异常。

这里发生了什么?我正在尝试在Clojure中为BaseX数据库编写一个插件;以上准确地表示了他们的插件加载机制如何工作以便为此问题提供SSCE。

1 个答案:

答案 0 :(得分:0)

BaseX使用的插件加载机制应该扩展为修改当前线程的上下文类加载器。对于给定的示例代码(在Jython中),这将类似于以下内容:

currentThread = java.lang.Thread.currentThread()
oldLoader = currentThread.getContextClassLoader()

currentThread.setContextClassLoader(customLoader)
try:
    cls = java.lang.Class.forName('aot_demo.JavaClass', True, customLoader)
finally:
    currentThread.setContextClassLoader(oldLoader)