类的卸载:由JVM自带的类加载器所加载的类,在JVM的生命周期中,始终不会被卸载。JVM本身会始终引用这些类加载器,而这些类加载器始终引用它们所加载的类的Class对象。所以说,这些Class对象始终是可触及的。
由用户自定义的类加载器所加载的类是可以被卸载的。

对于每一个Class对象,可以通过其getClassLoader()方法获得其类加载器的引用,所以Class对象内部有指向其类加载器的引用;而对于ClassLoader对象,在内部实现中,用一个Java集合来存放所已经加载过的类的引用。所以对于Class对象和ClassLoader对象之间是双向关联的。
一个类的实例总是引用代表这个类的Class对象,在Object类中定义了getClass()方法,这个方法返回该对象所代表类的Class对象的引用。此外,所有的Java类都有一个静态属性class,它引用代表这个类的Class对象。


下面,将上一篇博文 Java的自定义类加载器及JVM自带的类加载器之间的交互关系 中的示例类中的main方法进行修改:

Class objClass = loader1.loadClass("Sample");

System.out.println(objClass.hashCode());

Object obj = objClass.newInstance();
//5
loader1 = null;
objClass = null;
obj = null;
//8
loader1 = new MyClassLoader("loader1");
loader1.setPath("D:\\myapp\\serverlib\\");
objClass = loader1.loadClass("Sample");
System.out.println(objClass.hashCode());
obj = objClass.newInstance();
//10

输出结果:

java 卸载已加载的类 java类卸载条件_类卸载


下面我们来分析:

java 卸载已加载的类 java类卸载条件_加载_02


其中,loader1指向MyClassLoader对象

objClass指向代表Sample类的Class对象

obj指向Sample对象

类加载器与被该类加载器加载的Class对象之间相互关联,Sample对象指向代表Sample类的Class对象。其中,Class对象又指向方法区中Sample类的二进制数据结构。

当执行到8处时,三个引用变量都置为null,此时MyClassLoader对象结束生命周期,Sample类也会结束生命周期,代表Sample类的Class对象结束生命周期,Sample类在方法去的二进制数据结构也会被卸载。

当执行到10处时,都重新被创建加载,此时结构又如上图所示,不过这两次加载的Class对象并不是同一个,我们从输出结果可以看出hashCode值并不同。

如果同时对某一个类加载两次,如下:

Class clazz = loader1.loadClass("Sample");
Class clazz2 = loader1.loadClass("Sample");
System.out.println(clazz == clazz2);

输出结果为:true
同一个类的Class对象只有一个,当该类对象被加载后,就不会再去加载该类对应的Class对象,即使又执行了加载对象的操作。