一、當某個類被加載,鏈接和初始化後,它的生命週期就開始了。當表明這個類的Class對象再也不被引用,即不可觸及時,Class對象就會結束生命週期,這個類在方法區內的數據也會被卸載,從而結束這個類的生命週期。html
二、一個類什麼時候結束生命週期,取決於表明它的Class對象什麼時候結束生命週期。java
三、由Java虛擬機自帶的類加載器所加載的類,在虛擬機的生命週期中,始終不會被卸載。前面已經介紹過,Java虛擬機自帶的類加載器包括根加載器、擴展類加載器和系統加載器。Java虛擬機會始終引用這些類加載器,而這些類加載器則會始終引用它們所加載的Class對象,所以這些Class對象始終是可觸及的。jvm
四、由用戶自定義的類加載器所加載的類是能夠被卸載的。ide
五、運行以上程序時,Sample類由loader1加載。在類加載器的內部實現中,用一個Java集合來存放所加載類的引用。另外一方面,一個Class對象老是會引用它類加載器,調用Class對象的getClassLoader()方法,就會得到它的類加載器。因而可知,表明Sample類的Class實例與loader1之間爲雙向關聯關係。post
一個類的實例老是引用表明這個類的Class對象。在Object類中定義了getClass()方法,這個方法返回表明對象所屬的Class對象的引用。此外,全部的Java類都有一個靜態屬性class,它引用表明這個類的Class對象。this
六、類卸載的例子url
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:+TraceClassUnloadingspa
運行結果:htm
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
七、使用JVisualVM查看類的卸載對象
增長睡眠1分鐘(JVisualVM的使用參考JVisualVM監控本地Java進程)
而後使用JVisualVM觀察,能夠發現已經卸載類的總數爲1