原文地址:http://yhjhappy234.blog.163.com/blog/static/316328322011101413730764/?suggestedreading&wumiijava
咱们知道,当一个类被加载、链接和初始化以后,他的生命周期就开始了,当该类的class对象再也不被引用以后,该类的生命周期也就结束了,以后,该类会被类加载器卸载!app
咱们来看如下代码:jvm
package com.yhj.jvm.classloader.uninstall;< xmlnamespace prefix ="o" ns ="urn:schemas-microsoft-com:office:office" /> import com.yhj.jvm.classloader.user_defined.MyClassLoader; /** * @Described:类的卸载测试 * @author YHJ create at 2011-10-17 下午10:15:52 * @FileNmae com.yhj.jvm.classloader.uninstall.UninstallTest.java */ public class UninstallTest { /** * @param args * @throws ClassNotFoundException * @throws IllegalAccessException * @throws InstantiationException * @Author YHJ create at 2011-10-17 下午10:15:40 */ public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException { MyClassLoader classLoader1 = new MyClassLoader("classLoader1"); classLoader1.setClassPath("d:/jvmTest/server/"); Class<?> clazz = classLoader1.loadClass("TestCase1"); @SuppressWarnings("unused") Object obj = clazz.newInstance(); System.out.println("1:"+clazz.hashCode()); obj=null; System.out.println("2:"+clazz.hashCode()); classLoader1 = null; System.out.println("3:"+clazz.hashCode()); clazz = null; System.out.println("==========="); classLoader1 = new MyClassLoader("classLoader1"); classLoader1.setClassPath("d:/jvmTest/server/"); clazz = classLoader1.loadClass("TestCase1"); System.out.println("4:"+clazz.hashCode()); } }
如下代码先经过自定义类加载器classloader1加载类Testcase1,而后经过反射获取对应的obj对象,而后逐个将对应的对象设置为null以便于销毁对象,执行结果是是什么样子的呢?测试
咱们看到,前三次打印对象的spa
hashCode是一致的,说明是一个对象,可是最后一次却反生了变化,说明对象的地址变化了,也就是以前被卸载了(由于咱们知道,类的class对象在整个类加载过程当中只会被加载一次,只有一个实例对象,除非类被卸载了),而后又被加载了一次(加载了2次)。指针
咱们再来看这几个类的关系,咱们把它画成图,以下code
栈空间变量指向堆控件的数据,已经没什么争论,以前(上几篇文章)已经讨论过,loader1对象能够经过findClass方法拿到对应的TestCase1类,所以有一个指针指向TestCase1的Class实例,而TestCase1能够经过getClassLoader的方法拿到对应的类加载器,所以有一个指针指向MyClassloader的对象,TestCase1的实例obj经过getClass方法能够拿到对应的TestCase1实例。咱们这样能够看到,MyClassloader对象有2个引用指针,TestCase1有3个引用指针,当cloder1为null,objClass为null,obj为null的时候,系统还剩下MyClassloader和TestCase1的一个相互引用。server
咱们知道,只要class引用不在有,那么这个类就能够回收了,指向方法区的就不看了,可是MyClassloader和TestCase1每一个还有一个引用么人这两个其实已经没有其余引用了,可是从程序结果咱们能够看到,类已经被卸载了,说明JVM已经对这种双向引用也作了卸载和回收操做,可是按照咱们以前的理论,只有当引用计数器为0的时候才会被回收,这又如何理解呢?这个咱们会在下面的文章,深刻理解JVM—垃圾回收机制作详细介绍,敬请期待哦!xml