当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步时,引用变量与对象之间的引用关系如图:

java 卸载 java 卸载类_java 卸载

从上图可以看出,loader1变量和obj变量间接引用代表Sample类的Class对象,而objClass变量则直接引用它。

当程序执行完第8步时,所有的应用变量都置为null,此时Sample对象结束生命周期,MyClassLoader对象结束生命周期,代表Sample类的Class对象也结束生命周期,Sample类在方法区内的二进制数据被卸载。

当程序执行完第11步时,Sample类又重新被加载,在Java虚拟机的堆区会生成一个新的代表Sample类的Class实例。


从以上打印结果可以看出,程序两次打印objClass变量引用的Class对象的哈希码,得到的数值不同,因此objClass变量两次引用不同的class对象,可见在Java虚拟机的生命周期中,对Sample类先后加载了两次。