读书人

封锁URLClassLoader打开的jar包

发布时间: 2012-12-26 14:39:29 作者: rapoo

关闭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可能不能运行。

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中的ucpObject 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内部的jarLoaderMethod 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());}}}}
? 1 楼 feng1018 2011-07-05 你用的是jdk版本是?我的是1.5调试时,没用 m = jarLoader.getClass().getDeclaredMethod("ensureOpen");
这个方法,我去掉这个方法,可以进行jarfile.close,但是对应的jar文件还是不能进行修改/删除操作啊。 2 楼 feng1018 2011-07-05 大侠,你怎么用的,我在程序删除web-inf/lib下面的包,删除不了
调用 closeJAR()方法后,获取web-inf/lib下面的jar包,直接file.delete()但删除不了,直接在电脑里面删除也删除不了,只有关了tomcat才能删除(我使用tomcat)
private void closeJAR() throws Exception{
URLClassLoader loader = (URLClassLoader) getClass().getClassLoader();
//((WebappClassLoader)loader).closeJARs(true);
// 查找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", URL.class);
m.setAccessible(true);
Object jarLoader = m.invoke(ucpObj, list[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", URL.class);
m.setAccessible(true);
JarFile jf = (JarFile)m.invoke(jarLoader, list[i]);
// 释放jarLoader中的jar文件
jf.close();
jf = null;
System.out.println("release jar: "+jf.getName());
}
}
} 3 楼 weifly 2011-07-10 feng1018 写道你用的是jdk版本是?我的是1.5调试时,没用 m = jarLoader.getClass().getDeclaredMethod("ensureOpen");
这个方法,我去掉这个方法,可以进行jarfile.close,但是对应的jar文件还是不能进行修改/删除操作啊。

tomcat的webClassLoader自己来管理这些被打开的jar文件,没有用父类(URLClassloader)中的变量,因此你调用URLClassloader中的方法关闭jar文件是不管用的,应该调用tomcat的WebClassloader中的方法

读书人网 >编程

热点推荐