7

我有一个小问题。我学习 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)

你知道我该如何解决这个问题吗?

4

3 回答 3

3
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();

(通常的堆栈溢出免责声明与其说是编译有关。)

您需要添加错误处理来品尝。URLClassLoader.newInstance为 . 添加了一些细化URLClassLoaderClass.newInstance已经完全破坏了异常处理,应该避免。

于 2012-03-13T21:03:53.943 回答
3

上面的答案都是错误的,他们不了解根本问题。您的 main 指的是由一个类加载器加载的 Stack 类。您的 urlclassloader 正在尝试加载具有相同名称的类。您不能将加载的内容转换为引用的内容,因为它们不一样,它们属于不同的类加载器。您可以打印每个的 has 代码以查看它们是否不同。相等测试还将显示 cclass 引用不同。您的问题可能是因为可以找到 sstack 引用的依赖类,这将导致NoClassDefErrors等。您的 main 可能会因 classcastexception 而失败。

于 2012-03-15T11:08:32.553 回答
0

您不能在代码中按名称引用动态加载的类型,因为这必须在编译时解决。您需要newInstance()使用ClassloadClass().

于 2012-03-13T21:02:02.577 回答