一尘不染

卸载Java中的类?

java

我有一个自定义类加载器,以便桌面应用程序可以动态地开始从需要与之交谈的AppServer加载类。我们这样做是因为这样做所需的jar数量太可笑了(如果我们想发货的话)。如果我们没有在运行时从AppServer库动态加载类,就会遇到版本问题。

现在,我遇到一个问题,需要与两个不同的AppServer进行交谈,发现根据我首先加载的类,我可能会严重损坏…是否有任何方法可以在不杀死JVM的情况下强制卸载该类?

希望这有意义


阅读 1008

收藏
2020-02-27

共1个答案

一尘不染

可以卸载类的唯一方法是,如果使用的类加载器是垃圾回收。这意味着对每个单个类的引用以及对类加载器的引用都需要遵循dodo的方式。

一种可能的解决方案是为每个jar文件提供一个Classloader,为每个AppServer提供一个Classloader,以将类的实际加载委托给特定的Jar类加载器。这样,您可以为每个App服务器指向jar文件的不同版本。

不过,这并非微不足道。OSGi平台努力做到这一点,因为每个捆绑软件都有一个不同的类加载器,并且依赖项由平台解决。也许一个好的解决方案就是看看它。

如果您不想使用OSGI,则一种可能的实现方式是为每个JAR文件使用一个JarClassloader类实例。

并创建一个扩展Classloader的新的MultiClassloader类。此类在内部将具有JarClassloaders的数组(或List),并且在defineClass()方法中将迭代所有内部类加载器,直到找到定义或抛出NoClassDefFoundException为止。可以提供几个访问器方法以将新的JarClassloaders添加到该类。网上有多种针对MultiClassLoader的实现,因此您甚至不需要编写自己的实现。

如果为服务器的每个连接实例化MultiClassloader,则原则上每个服务器都可能使用同一类的不同版本。

我在一个项目中使用了MultiClassloader的想法,其中包含用户定义脚本的类必须从内存中加载和卸载,并且运行良好。

2020-02-27