当Sample类被加载、连接和初始化后,它的生命周期就开始了。当代表Sample类的Class对象不再被引用,即不可触及时,Class对象就会结束生命周期,Sample类在方法区内的数据也会被卸载,从而结束Sample类的生命周期。由此可见,一个类何时结束生命周期,取决于代表它的Class对象何时结束生命周期。
由Java虚拟机自带的类加载器所加载的类,在虚拟机的生命周期中,始终不会被卸载。Java虚拟机自带的类加载器包括根类加载器、扩展类加载器和系统类加载器。Java虚拟机自带的类加载器包括根类加载器、扩展类加载器和系统类加载器。Java虚拟机本身会始终引用这些类加载器,而这些类加载器则会始终引用它们所加载的类的Class对象,因此这些Class对象始终是可触及的。
由用户自定义的类加载器所加载的类是可以被卸载的。
public static void main(String[] aregs) throws Exception
{
MyClassLoader loader1 = new MyClassLoader("loader1"); //1
loader1.setPath("d:\\myapp\\serverlib\\"); //2
Class objClass = loader1.loadClass("Sample"); //3
System.out.println("objClass's hashCode is "+objClass.hashCode()); //4
Object object = objClass.newInstance(); //5
loader1=null; //6
objClass = null; //7
object = null; //8
loader1 = new MyClassLoader("loader1"); //9
loader1.setPath("d:\\myapp\\serverlib\\"); //10
objClass=loader1.loadClass("Sample"); //11
System.out.println("objClass's hashCode is "+objClass.hashCode()); //12
}
输出结果:
objClass's hashCode is 118352462
Sample is loaded by: loader1
Dog is loaded by :loader1
objClass's hashCode is 1442407170
运行以上程序时,Sample类由loader1加载。在类加载器的内部实现中,用一个Java集合来存放所加载类的引用。另一方面,一个Class对象总是会引用它的类加载器,调用Class对象的getClassLoader()方法,就能获得它的类加载器。由此可见,代表Sample类的Class实例与loader1之间为双向关联关系。
一个类的实例总是引用代表这个类的Class对象。在Object类中定义了getClass()方法,这个方法返回代表对象所属类的Class对象的引用。此外,所有的Java类都有一个静态属性class,它引用代表这个类的Class对象,例如:
当程序执行完第5步时,引用变量与对象之间的引用关系如图:
从上图可以看出,loader1变量和obj变量间接引用代表Sample类的Class对象,而objClass变量则直接引用它。
当程序执行完第8步时,所有的应用变量都置为null,此时Sample对象结束生命周期,MyClassLoader对象结束生命周期,代表Sample类的Class对象也结束生命周期,Sample类在方法区内的二进制数据被卸载。
当程序执行完第11步时,Sample类又重新被加载,在Java虚拟机的堆区会生成一个新的代表Sample类的Class实例。
从以上打印结果可以看出,程序两次打印objClass变量引用的Class对象的哈希码,得到的数值不同,因此objClass变量两次引用不同的class对象,可见在Java虚拟机的生命周期中,对Sample类先后加载了两次。