我有小问题。我学习java SE并找到类ClassLoader。我尝试在下面的代码中使用它: 我正在尝试使用URLClassLoader在运行时动态加载类。
URLClassLoader urlcl = new URLClassLoader(new URL[] {new URL("file:///I:/Studia/PW/Sem6/_repozytorium/workspace/Test/testJavaLoader.jar")});
Class<?> classS = urlcl.loadClass("michal.collection.Stack");
for(Method field: classS.getMethods()) {
System.out.println(field.getName());
}
Object object = classS.newInstance();
michal.collection.Stack new_name = (michal.collection.Stack) object;
java虚拟机看不到我的类,我得到以下异常:
Exception in thread "main" java.lang.Error: Unresolved compilation problems: michal cannot be resolved to a type michal cannot be resolved to a type at Main.main(Main.java:62)
你知道如何解决这个问题吗?
答案 0 :(得分:4)
Class<?> classS = urlcl.loadClass("michal.collection.Stack");
[...]
Object object = classS.newInstance();
michal.collection.Stack new_name = (michal.collection.Stack) object;
所以你试图动态加载一个类然后你静态地引用它。如果您已经可以静态链接到它,那么它已加载,您无法再次加载它。您需要通过反射来访问这些方法。
您通常要做的是让加载的类实现父类加载器的接口。在创建实例(通常只是一个实例)之后,您可以通过具有接口类型的引用来引用它。
public interface Stack {
[...]
}
[...]
URLClassLoader urlcl = URLClassLoader.newInstance(new URL[] {
new URL(
"file:///I:/Studia/PW/Sem6/_repozytorium/workspace/Test/testJavaLoader.jar"
)
});
Class<?> clazz = urlcl.loadClass("michal.collection.StackImpl");
Class<? extends Stack> stackClass = clazz.asSubclass(Stack.class);
Constructor<? extends Stack> ctor = stackClass.getConstructor();
Stack stack = ctor.newInstance();
(通常的Stack Overflow免责声明与编译无关。)
您需要添加错误处理才能尝试。 URLClassLoader.newInstance
为URLClassLoader
添加了一些改进。 Class.newInstance
完全破坏了异常处理,应该避免使用。
答案 1 :(得分:0)
您无法在代码中按名称引用动态加载的类型,因为必须在编译时解析。您需要使用从newInstance()
返回的Class
对象的loadClass()
函数。
答案 2 :(得分:0)
以上答案都是错误的,他们不了解根本问题。您的main指的是由一个类加载器加载的Stack类。您的urlclassloader正在尝试加载具有相同名称的类。你不能将加载到被引用因为它们不相同,它们属于不同的类加载器。您可以打印每个的代码,看它们是不同的。相等测试还将显示cclass引用不同。您的问题可能是因为可以找到sstack引用的依赖类,这将在NoClassDefErrors等中进行查询。您的main可能会因classcastexception而失败。