【发布时间】:2017-02-25 09:01:54
【问题描述】:
我正在尝试从动态加载的 jar 文件中加载类。目前,我可以加载一个不扩展为根类 Object 以外的类的类,但我无法加载扩展自定义类的类。我的问题的详细描述如下。
在项目A中,有一个类A,如:
class A {
}
在项目 B 中,有一个 B1 类和一个 B2 类,例如:
class B1 {
}
class B2 extends A {
}
我将项目 A 导出到 a.jar,将项目 B 导出到 b.jar。测试代码为:
File f = new File("Z:\\Jars\\b.jar");
URLClassLoader urlClassLoader = new URLClassLoader(new URL[] { f.toURL() },
System.class.getClassLoader());
Class<?> classB1 = (Class<?>) urlClassLoader.loadClass("b.B1");
System.out.println("B1 is loaded");
Class<?> classB2 = (Class<?>) urlClassLoader.loadClass("b.B2");
System.out.println("B2 is loaded");
类B1加载成功,但是由于类a.A未找到异常导致B2类没有加载:
java -cp a.jar;b.jar; a.Main
B1 is loaded
Exception in thread "main" java.lang.NoClassDefFoundError: a/A
Caused by: java.lang.ClassNotFoundException: a.A
at java.net.URLClassLoader.findClass(Unknown Source)
at java.lang.ClassLoader.loadClass(Unknown Source)
at java.lang.ClassLoader.loadClass(Unknown Source)
... 12 more
我该如何解决这个问题? 谢谢。
附言。包含 a.jar 后,可以加载 B2 类,但无法通过以下测试:
Class<A> classB2 = (Class<A>) urlClassLoader.loadClass("b.B2");
A ab2 = classB2.newInstance();
第一行没问题,但第二行触发了以下异常:
Exception in thread "main" java.lang.ClassCastException: b.B2 cannot be cast to a.A
at a.Main.main(Main.java:22)
B2 应该是 A 实例,知道为什么会这样吗?
【问题讨论】:
-
如何在不包含 A 类的情况下构建 jar b.jar?您是从 b.jar 中手动删除 A.class 吗?
-
我用的是Eclipse,我把B项目设置为依赖A项目,这样B项目就可以构建,并且可以导出一个jar文件。对于上面的演示案例,我的问题通过包含 a.jar 得到了解决,可以加载 B2 类并将其强制转换为 A。
-
stackoverflow.com/questions/2591779/cast-across-classloader的第一个回答说不同类加载器加载的同一个类的类不能相互转换。
-
@DeepParticle 正确,请参阅我的回答,了解您可以做些什么。
标签: java jar jvm classloader dynamic-class-loaders