`
weifly
  • 浏览: 240370 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

关闭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中的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());
			}
		}
	}
}
 
分享到:
评论
5 楼 liang5233145 2016-03-25  
1.7版本,有个close()方法,但是1.7之前没有关闭方法,,这些天也遇到这个问题,因为各种原因,要用1.6版本的jdk,经测试,楼主方法没问题,但是tomact下,这句 URLClassLoader loader = new URLClassLoader(urls);  应改为      URLClassLoader loader = new URLClassLoader(urls,Thread.currentThread().getContextClassLoader());  报错的同学,可以试着改下这句。
4 楼 di1984HIT 2013-05-22  
1.7提供close方法了。比较方便啊
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中的方法
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());  
            }  
        }  
}
1 楼 feng1018 2011-07-05  
你用的是jdk版本是?我的是1.5调试时,没用 m = jarLoader.getClass().getDeclaredMethod("ensureOpen");  
这个方法,我去掉这个方法,可以进行jarfile.close,但是对应的jar文件还是不能进行修改/删除操作啊。

相关推荐

    Java源文件读取JAR包文件.rar

    在Java编程中,有时我们需要在运行时从JAR(Java ...通过以上步骤,你可以在Java源文件中成功地读取并处理JAR包文件中的内容。记住,这只是一个基础的实现,根据具体需求,你可能需要添加更多的错误处理和优化措施。

    ClassLoaderTest:测试如何正确关闭类加载器,以便可以关闭基础jar文件上的文件句柄

    关闭类加载器的一个常见问题是,当我们动态加载了jar文件中的类后,jar文件会被一个JarFile对象打开,这个对象持有对文件的句柄。如果不正确关闭,这个句柄将持续占用,导致资源泄漏。因此,我们需要在适当的时候...

    从JAR文件中读取文本数据

    1. **打开JAR文件**:创建`JarFile`对象,传入JAR文件的路径。例如: ```java JarFile jar = new JarFile("path_to_your_jar.jar"); ``` 2. **获取条目**:使用`getJarEntry`方法获取特定路径的条目,该条目代表...

    lassLoader的关系以及如何防止ClassLoader内存泄漏

    5. 文件句柄或网络连接:类加载器可能打开文件或保持网络连接,如果在加载完成后未关闭,也会导致资源泄露。务必在类加载器不再使用时关闭这些资源。 总之,理解ClassLoader的工作原理及其关系对于优化Java应用程序...

    JAVA7新特性介绍(官方公布)中文版[参照].pdf

    Java 7引入了一个新的语法结构——try-with-resources,使得资源的打开和关闭更加简洁和安全,减少了可能出现的资源泄露。 10. **多catch块**: 开发者可以在一个catch块中捕获多个异常类型,简化了异常处理代码...

Global site tag (gtag.js) - Google Analytics