我正在编写一个服务器程序,用于运行API的单元测试 (显示大量信息并提供Web访问来控制/监控整个事物)...
该API 在编译期间为服务器所知,并作为JAR提供.
为了能够比较不同版本 API的单元测试结果(无需重新启动服务器),我希望能够卸载API的"当前"版本,并重新加载较新版本(或较旧的版本) .
我不想使用URLClassLoader并按名称
(使用 getDeclaredMethod("someMethod")
)调用每个方法,
因为服务器在很大程度上依赖于API,并且以这种脏的方式"包装"每个方法调用会很复杂.
我在想:由于JAR 的所有版本的所有接口都是相同的,我不能通过某种方式重新加载其他版本的JAR(没有那个by-name-invokation?).
注意:我使用的是最新的Java SE(6)和Java EE(5).
如果您认为,我想要达到的目标是不可能的,请提出"解决方法"或不同的概念.
我想如果你加载一个类使用
Class.forName(clsname, init, classloader);
(这里是Javadoc)您将获得给定类加载器提供的类的实例.由于该类而加载的所有内容也将通过相同的类加载器加载.
只要你非常小心从这一点开始实例化的对象(允许GC),你应该能够重新加载不同的版本.我之前用Java 1.3做了一次,它花了很多调试,但最后我有一个"bootstrap"应用程序,它Runnable
按名称加载一个类,并且能够通过实例化一个新的类加载器来"软重启"网址然后再去.
您可以以编程方式修改类路径以反映您的JAR更改。这是我的方法:
URLClassLoader urlClassLoader = (URLClassLoader) ClassLoader.getSystemClassLoader(); Method m = URLClassLoader.class.getDeclaredMethod("addURL", new Class[]{URL.class}); m.setAccessible(true); m.invoke(urlClassLoader, jarFile.toURI().toURL()); String cp = System.getProperty("java.class.path"); if (cp != null) { cp += File.pathSeparatorChar + jarFile.getCanonicalPath(); } else { cp = jarFile.toURI().getPath(); } System.setProperty("java.class.path", cp);
其中jarFile是要使用/覆盖的jar版本。