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可能不能运行。
01 02 03 04 05 06 07 08 09 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 | 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包。