我需要为应用程序的某些部分向现有应用程序添加插件功能。我希望能够在运行时添加一个 jar,并且应用程序应该能够从 jar 加载一个类而无需重新启动应用程序。到现在为止还挺好。我使用 URLClassLoader 在网上找到了一些示例,它工作正常。
我还希望能够在 jar 的更新版本可用时重新加载相同的类。我再次找到了一些示例,据我所知,实现这一目标的关键是我需要为每个新负载使用一个新的类加载器实例。
我写了一些示例代码,但遇到了 NullPointerException。首先给大家看一下代码:
package test.misc;
import java.io.File;
import java.net.URL;
import java.net.URLClassLoader;
import plugin.misc.IPlugin;
public class TestJarLoading {
public static void main(String[] args) {
IPlugin plugin = null;
while(true) {
try {
File file = new File("C:\\plugins\\test.jar");
String classToLoad = "jartest.TestPlugin";
URL jarUrl = new URL("jar", "","file:" + file.getAbsolutePath()+"!/");
URLClassLoader cl = new URLClassLoader(new URL[] {jarUrl}, TestJarLoading.class.getClassLoader());
Class loadedClass = cl.loadClass(classToLoad);
plugin = (IPlugin) loadedClass.newInstance();
plugin.doProc();
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
Thread.sleep(30000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
}
IPlugin 是一个简单的接口,只有一个方法 doProc:
public interface IPlugin {
void doProc();
}
jartest.TestPlugin 是这个接口的一个实现,doProc 只是打印出一些语句。
现在,我将 jartest.TestPlugin 类打包到一个名为 test.jar 的 jar 中,并将其放在 C:\plugins 下并运行此代码。第一次迭代运行顺利,类加载没有问题。
当程序执行 sleep 语句时,我将 C:\plugins\test.jar 替换为包含同一类的更新版本的新 jar,并等待下一次 while 迭代。现在这是我不明白的。有时更新的类会重新加载而没有问题,即下一次迭代运行良好。但有时,我看到抛出异常:
java.lang.NullPointerException
at java.io.FilterInputStream.close(FilterInputStream.java:155)
at sun.net.www.protocol.jar.JarURLConnection$JarURLInputStream.close(JarURLConnection.java:90)
at sun.misc.Resource.getBytes(Resource.java:137)
at java.net.URLClassLoader.defineClass(URLClassLoader.java:256)
at java.net.URLClassLoader.access$000(URLClassLoader.java:56)
at java.net.URLClassLoader$1.run(URLClassLoader.java:195)
at java.security.AccessController.doPrivileged(Native Method)
at java.net.URLClassLoader.findClass(URLClassLoader.java:188)
at java.lang.ClassLoader.loadClass(ClassLoader.java:307)
at java.lang.ClassLoader.loadClass(ClassLoader.java:252)
at test.misc.TestJarLoading.main(TestJarLoading.java:22)
我已经在网上搜索并挠了挠头,但无法真正得出关于为什么会引发此异常的任何结论 - 只是有时,并非总是如此。
我需要你的经验和专业知识来理解这一点。这段代码有什么问题?请帮忙!!
如果您需要更多信息,请告诉我。感谢您的关注!