我的项目结构如下(当然非常简化):
因此,在lib-ext
下,我每天从Jenkins服务器下载2个jar文件“ jar1和jar2”,以供我的程序检查,我需要从“ jar1”中获取一个文件,称为“ Class2Bloaded”。
问题在于此文件实现了在“ jar2”中找到的接口,我们称其为“ Dependency”
我想做的是,从我的类在src“ ClassThatLoads.java”下,加载'Class2Bloaded.class'并告诉类加载器查看'jar2'以搜索实现接口“ Dependency.class”
到目前为止,我的代码(省略了异常处理):
//Create the URL pointing to Jar1
private URL getJarUrl(JarFile jarFile)
{
return new File(jarFile.getName()).toURI().toURL();
}
URL jar1Url = getJarUrl(jar1);
ClassLoader jar1classLoader = new URLClassLoader(new URL[] { jar1Url });
Class<?> Class2Bloaded = Class.forName(fullClassName, false, jar1classLoader );
所以问题发生在
Class.forName
调用内,因为我要加载的类实现了jar 2中的接口。Exception in thread "main" java.lang.NoClassDefFoundError: com/packagewithinJar2/Dependency
因此,最终我准备了另一个指向“ jar2”的类加载器,甚至获得了所需的实际接口:
URL jar2Url = getJarUrl(jar2);
ClassLoader jar2classLoader = new URLClassLoader(new URL[] { jar2Url });
Class<?> Interface2Bloaded = Class.forName(fullClassName, false, jar2classLoader );
在第二种情况下,“ fullClassName”是“ Class2Bloaded”所依赖的接口的标准名称。
只是我无法在
ClassLoader
的javadocs中找到任何东西,允许我为依赖项“注入”其他类加载器。希望我的解释清楚。
最佳答案
首先要做的是将jar2添加到URLClassLoader读取的jar列表中:
ClassLoader jarclassLoader = new URLClassLoader(new URL[] { jar1Url, jar2Url });
但是通常要做的是从头开始在您的类路径上添加jar1和jar2。
为此,您将使用java可执行文件的-cp参数。
例如,如果将类编译到bin目录中:
java -cp libext/jar1.jar:libext/jar2.jar:bin ClassThatLoads
这样,您可以在自己的Java源代码中使用无缝的类,并摆脱麻烦的加载部分:
public class ClassThatLoads {
public static void main(String[] args) {
Class2Bloaded stuff = new Class2Bloaded();
//use stuff from here...
}
}