JVM 類的卸載


類的卸載:由JVM自帶的類加載器所加載的類,在JVM的生命周期中,始終不會被卸載。JVM本身會始終引用這些類加載器,而這些類加載器始終引用它們所加載的類的Class對象。所以說,這些Class對象始終是可觸及的。 

由用戶自定義的類加載器所加載的類是可以被卸載的。
 
當類被加載,連接和初始化后,它的生命周期就開始了。當 代表類的Class對象不在被引用時,即不可觸及時,Class對象就會結束生命周期,類在方法區內的數據也會被卸載,從而結束類的生命周期。由此可見,一個類何時結束生命周期,取決於代表它的Class對象何時結束生命周期。由Java虛擬機自帶的類加載器所加載的類,在虛擬機的生命周期中,始終不會被卸載。包括根類加載器,擴展類加載器和系統類加載器,Java虛擬機本身會始終引用這些類加載器,而這些類加載器則會始終引用這些類加載器,而這些類加載器則會始終引用它們所加載的類的Class對象,因此這些Class對象始終是可觸及的。
如果程序運行過程中,將上圖左側三個引用變量都置為null,此時Sample對象結束生命周期,MyClassLoader對象結束生命周期,代表Sample類的Class對象也結束生命周期,Sample類在方法區內的二進制數據被卸載。
當再次有需要時,會檢查Sample類的Class對象是否存在,如果存在會直接使用,不再重新加載;如果不存在Sample類會被重新加載,在Java虛擬機的堆區會生成一個新的代表Sample類的Class實例(可以通過哈希碼查看是否是同一個實例)。
 
 
通過虛擬機參數,可以查看類的加載與卸載過程
  • -verbose:class 跟蹤類的加載和卸載
  • -XX:+TraceClassLoading 跟蹤類的加載
  • -XX:+TraceClassUnloading 跟蹤類的卸載

 

(1)系統類加載器始終是不會被卸載的,示例如下:
添加虛擬機參數:-verbose:class
 
System.out.println("第一次開始加載Sample類");
Class<?> clazz = ClassLoader.getSystemClassLoader().loadClass(Sample.class.getName());
Object obj = clazz.newInstance();
System.out.println(clazz.hashCode());    // 2018699554
obj = null;
clazz = null;
System.gc();
System.out.println("第二次開始加載Sample類");
clazz = ClassLoader.getSystemClassLoader().loadClass(Sample.class.getName());
System.out.println(clazz.hashCode());    // 2018699554
Thread.sleep(2000);
System.out.println("執行結束....");

控制台輸出:

此例也說明,由系統類加載器加載的類不會被卸載,並且只加載一次,Class對象也只有一個

 

(2)用戶自定義類加載器可以卸載,示例如下
添加虛擬機參數:-verbose:class
System.out.println("開始加載Sample類");
MyClassLoader myClassLoader = new MyClassLoader();
Class<?> clazz = myClassLoader.findClass(Sample.class.getName());
Object obj = clazz.newInstance();
// 當代表類的Class對象不在被引用時,Class對象就會結束生命周期,類在方法區內的數據也會被卸載
obj = null;
clazz = null;
myClassLoader = null;
System.gc();
Thread.sleep(2000);
System.out.println("執行結束....");

控制台輸出:

 
 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM