1、當某個類被加載,連接和初始化后,它的生命周期就開始了。當代表這個類的Class對象不再被引用,即不可觸及時,Class對象就會結束生命周期,這個類在方法區內的數據也會被卸載,從而結束這個類的生命周期。
2、一個類何時結束生命周期,取決於代表它的Class對象何時結束生命周期。
3、由Java虛擬機自帶的類加載器所加載的類,在虛擬機的生命周期中,始終不會被卸載。前面已經介紹過,Java虛擬機自帶的類加載器包括根加載器、擴展類加載器和系統加載器。Java虛擬機會始終引用這些類加載器,而這些類加載器則會始終引用它們所加載的Class對象,因此這些Class對象始終是可觸及的。
4、由用戶自定義的類加載器所加載的類是可以被卸載的。
5、運行以上程序時,Sample類由loader1加載。在類加載器的內部實現中,用一個Java集合來存放所加載類的引用。另一方面,一個Class對象總是會引用它類加載器,調用Class對象的getClassLoader()方法,就會獲得它的類加載器。由此可見,代表Sample類的Class實例與loader1之間為雙向關聯關系。
一個類的實例總是引用代表這個類的Class對象。在Object類中定義了getClass()方法,這個方法返回代表對象所屬的Class對象的引用。此外,所有的Java類都有一個靜態屬性class,它引用代表這個類的Class對象。
6、類卸載的例子
public class MyTest161 extends ClassLoader{
private String className;
//目錄
private String path;
private final String fileExtension = ".class";
public MyTest161(String classLoadName){
super(); //將系統類加載器當做該類加載器的父加載器
this.className = classLoadName;
}
public MyTest161(ClassLoader parent, String classLoadName){
super(parent); //顯示指定該類加載器的父加載器器
this.className = classLoadName;
}
public void setPath(String path) {
this.path = path;
}
@Override
public String toString() {
return "[" + this.className + "]";
}
@Override
protected Class<?> findClass(String clasName) throws ClassNotFoundException {
System.out.println("findClass invoked:" + clasName);
System.out.println("class loader name: " + this.className);
byte[] data = this.loadClassData(clasName);
return this.defineClass(clasName,data, 0, data.length);
}
private byte[] loadClassData(String className){
InputStream is = null;
byte[] data = null;
ByteArrayOutputStream baos = null;
try{
className = className.replace(".","//");
//System.out.println("className:" +this.className);
is = new FileInputStream(new File(this.path + className + this.fileExtension));
baos = new ByteArrayOutputStream();
int ch = 0;
while ( -1 != (ch = is.read())){
baos.write(ch);
}
data = baos.toByteArray();
}catch (Exception ex){
ex.printStackTrace();
}finally {
try {
is.close();
baos.close();
}catch (Exception ex){
ex.printStackTrace();
}
}
return data;
}
public static void main(String[] args) throws Exception{
MyTest161 loader1 = new MyTest161("loader1");
loader1.setPath("D:/temp/");
Class<?> clazz = loader1.loadClass("com.example.jvm.classloader.MyTest1"); //
System.out.println("class:" + clazz.hashCode());
Object object = clazz.newInstance();
System.out.println(object);
loader1 = null;
clazz = null;
object = null;
System.gc(); // 垃圾回收。實際開發不會用
System.out.println();
loader1 = new MyTest161("loader1");
loader1.setPath("D:/temp/");
clazz = loader1.loadClass("com.example.jvm.classloader.MyTest1"); //
System.out.println("class:" + clazz.hashCode());
object = clazz.newInstance();
System.out.println(object);
System.out.println();
}
}
設置JVM參數 -XX:+TraceClassUnloading

運行結果:
findClass invoked:com.example.jvm.classloader.MyTest1 class loader name: loader1 class:21685669 com.example.jvm.classloader.MyTest1@7f31245a findClass invoked:com.example.jvm.classloader.MyTest1 class loader name: loader1 class:1173230247 [Unloading class com.example.jvm.classloader.MyTest1 0x0000000100061028] com.example.jvm.classloader.MyTest1@330bedb4
7、使用JVisualVM查看類的卸載
增加睡眠1分鍾(JVisualVM的使用參考JVisualVM監控本地Java進程)

然后使用JVisualVM觀察,可以發現已經卸載類的總數為1

