卸载URLClassLoader加载的Jar包

URLClassLoader可以让开发者定义classloader,从jar文件或目录加载类文件。当URLClassloader引用的是一个jar文件时,用户是不能自己关闭这个被打开的jar文件的,有时候这真是一个大问题。tomcat中可以动态的删除一个web应用,那他是怎么关闭打开的jar文件呢?这个web应用可以有自己的jar包,tomcat的也必须要打开这些jar包的。

tomcat的webClassLoader自己来管理这些被打开的jar文件,并在classloader中提供了closeJars方法,这样做真的是很明智的。jetty服务器中的WebAppClassLoader没有tomcat中的classLoader那么智能。

卸载URLClassLoader加载的Jar包

下面的类也可以达到关闭jar文件的功能,通过java中反射机制强行的关闭被打开的JarFile文件,但是这么做的话就依赖于JDK中的URLClassLoader的实现了,对于IBM或WEBLogic中的JDK可能不能运行。

import java.io.File;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.jar.JarFile;

public class TestClassLoader {

    public static void main(String[] args) throws Exception {
        File jar = new File("d:testcommons-lang-2.2.jar");
        URL[] urls = new URL[]{jar.toURI().toURL()};
        URLClassLoader loader = new URLClassLoader(urls);

        Class<?> cls = loader.loadClass("org.apache.commons.lang.StringUtils");
        System.out.println(cls.getName());

        close(loader);
    }

    public static void close(URLClassLoader loader) throws Exception {
        // 查找URLClassLoader中的ucp
        Field ucpField = URLClassLoader.class.getDeclaredField("ucp");
        ucpField.setAccessible(true);
        Object ucpObj = ucpField.get(loader);
        URL[] list = loader.getURLs();
        for(int i=0;i<list.length;i++) {
            // 获得ucp内部的jarLoader
            Method m = ucpObj.getClass().getDeclaredMethod("getLoader", int.class);
            m.setAccessible(true);
            Object jarLoader = m.invoke(ucpObj, i);
            String clsName = jarLoader.getClass().getName();
            if(clsName.indexOf("JarLoader")!=-1) {
                m = jarLoader.getClass().getDeclaredMethod("ensureOpen");
                m.setAccessible(true);
                m.invoke(jarLoader);
                m = jarLoader.getClass().getDeclaredMethod("getJarFile");
                m.setAccessible(true);
                JarFile jf = (JarFile)m.invoke(jarLoader);
                // 释放jarLoader中的jar文件,此处可根据jar包绝对路径判断哪些可以关闭
                // TODO
                jf.close();
                System.out.println("release jar: "+jf.getName());
            }
        }
    }
}

注意:TODO处可进行判断,筛选自己需要卸载的JAR包。

发表评论