yanqiufang 发表于 2017-2-28 10:47:00

关闭URLClassLoader打开的jar包

  URLClassLoader可以让开发者定义classloader,从jar文件或目录加载类文件。当URLClassloader引用的是一 个jar文件时,用户是不能自己关闭这个被打开的jar文件的,有时候这真是一个大问题。tomcat中可以动态的删除一个web应用,那他是怎么关闭打 开的jar文件呢?这个web应用可以有自己的jar包,tomcat的也必须要打开这些jar包的。
  tomcat的webClassLoader自己来管理这些被打开的jar文件,并在classloader中提供了closeJars方法,这样 做真的是很明智的。jetty服务器中的WebAppClassLoader没有tomcat中的classLoader那么智能,他们真的需要改进一 下。
  下面的类也可以达到关闭jar文件的功能,他通过java中反射机制强行的关闭被打开的JarFile文件,但是这么做的话就依赖于jdk中的URLClassLoader的实现了,对于IBM或WEBLogic中的jdk可能不能运行。



Java代码
[*]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:\\test\\commons-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());
[*]
[*]      // 查找URLClassLoader中的ucp
[*]      Object ucpObj = null;
[*]      Field ucpField = URLClassLoader.class.getDeclaredField("ucp");
[*]      ucpField.setAccessible(true);
[*]      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文件
[*]                jf.close();
[*]                System.out.println("release jar: "+jf.getName());
[*]            }
[*]      }
[*]    }
[*]}
  转载:

http://weifly.iteye.com/blog/941730, 但是在我机器上测试并没有发现存在锁Jar包的情况,这是为何呢?  说明一下:
  刚才自己debug了一下,找到原因,首先1.5下用这种方式应该关不掉,因为getJarFile这个无参方法是1.6提供的。
页: [1]
查看完整版本: 关闭URLClassLoader打开的jar包