java筆記--理解java類加載器以及ClassLoader類

類加載器概述:

  java類的加載是由虛擬機來完成的,虛擬機把描述類的Class文件加載到內存,並對數據進行校驗,解析和初始化,最終造成能被java虛擬機直接使用的java類型,這就是虛擬機的類加載機制.JVM中用來完成上述功能的具體實現就是類加載器.類加載器讀取.class字節碼文件將其轉換成java.lang.Class類的一個實例.每一個實例用來表示一個java類.經過該實例的newInstance()方法能夠建立出一個該類的對象.html

類的生命週期:

  類從加載到虛擬機內存到被從內存中釋放,經歷的生命週期以下:java

 

加載:"加載"是"類加載"過程的一個階段,此階段完成的功能是:安全

  經過類的全限定名來獲取定義此類的二進制字節流網絡

  將此二進制字節流所表明的靜態存儲結構轉化成方法區的運行時數據結構數據結構

  在內存中生成表明此類的java.lang.Class對象,做爲該類訪問入口.spa

驗證:鏈接階段第一步.驗證的目的是確保Class文件的字節流中信息符合虛擬機的要求,不會危害虛擬機安全,使得虛擬機免受惡意代碼的攻擊.大體完成如下四個校驗動做:代理

  文件格式驗證code

  源數據驗證htm

  字節碼驗證對象

  符號引用驗證

準備:鏈接階段第二步,正式爲類變量分配內存並設置變量的初始值.(僅包含類變量,不包含實例變量).  

解析:鏈接階段第三步,虛擬機將常量池中的符號引用替換爲直接引用,解析動做主要針對類或接口,字段,類方法,方法類型等等..

初始化:類的初始化是類加載過程的最後一步,在該階段,才真正意義上的開始執行類中定義的java程序代碼.該階段會執行類構造器.

使用:使用該類所提供的功能.

卸載:從內存中釋放.

 

獲取Class文件途徑:

  java類能夠動態被加載到內存,這是java的一大特色,也稱爲運行時綁定,或動態綁定.

     1.從ZIP包中讀取,很常見,最終成爲往後JAR,WAR,EAR格式的基礎.

     2.從網絡中獲取,這種場景典型的就是Applet.

     3.運行時計算生成,典型的情景就是java動態代理技術.

     4.從其餘文件中生成,典型場景是JSP應用,即由JSP文件生成對應的Class類.

 

java.lang.ClassLoader類概述:

  中文文檔中對ClassLoader類的定義以下:

   

   從文檔中對ClassLoader類的介紹能夠總結出這個類的做用就是根據一個指定的類的全限定名,找到對應的Class字節碼文件,而後加載它轉化成一個java.lang.Class類的一個實例.

類加載器的劃分:

   大部分java程序會使用如下3中系統提供的類加載器:

   啓動類加載器(Bootstrap ClassLoader):

    這個類加載器負責將<JAVA_HOME>\lib目錄下的類庫加載到虛擬機內存中,用來加載java的核心庫,此類加載器並不繼承於java.lang.ClassLoader,不能被java程序直接調用,代碼是使用C++編寫的.是虛擬機自身的一部分.

   擴展類加載器(Extendsion ClassLoader):
   
 這個類加載器負責加載<JAVA_HOME>\lib\ext目錄下的類庫,用來加載java的擴展庫,開發者能夠直接使用這個類加載器.

   應用程序類加載器(Application ClassLoader):

    這個類加載器負責加載用戶類路徑(CLASSPATH)下的類庫,通常咱們編寫的java類都是由這個類加載器加載,這個類加載器是CLassLoader中的getSystemClassLoader()方法的返回值,因此也稱爲系統類加載器.通常狀況下這就是系統默認的類加載器.

  除此以外,咱們還能夠加入本身定義的類加載器,以知足特殊的需求,須要繼承java.lang.ClassLoader類.

  類加載器之間的層次關係以下圖:

   

 

使用代碼觀察一下類加載器:

package com.wang.test;

public class TestClassLoader {

    public static void main(String[] args) {
        ClassLoader loader = TestClassLoader.class.getClassLoader();
        System.out.println(loader.toString());
        System.out.println(loader.getParent().toString());
        System.out.println(loader.getParent().getParent());
    }
}

 

  觀察打印結果:

sun.misc.Launcher$AppClassLoader@500c05c2
sun.misc.Launcher$ExtClassLoader@454e2c9c
null

  第一行打印的是應用程序類加載器(默認加載器),第二行打印的是其父類加載器,擴展類加載器,按照咱們的想法第三行應該打印啓動類加載器的,這裏卻返回的null,緣由是getParent(),返回時null的話,就默認使用啓動類加載器做爲父加載器.

 類加載器的雙親委派模型:

  雙親委派模型是一種組織類加載器之間關係的一種規範,他的工做原理是:若是一個類加載器收到了類加載的請求,它不會本身去嘗試加載這個類,而是把這個請求委派給父類加載器去完成,這樣層層遞進,最終全部的加載請求都被傳到最頂層的啓動類加載器中,只有當父類加載器沒法完成這個加載請求(它的搜索範圍內沒有找到所需的類)時,纔會交給子類加載器去嘗試加載.

  這樣的好處是:java類隨着它的類加載器一塊兒具有了帶有優先級的層次關係.這是十分必要的,好比java.langObject,它存放在\jre\lib\rt.jar中,它是全部java類的父類,所以不管哪一個類加載都要加載這個類,最終全部的加載請求都彙總到頂層的啓動類加載器中,所以Object類會由啓動類加載器來加載,因此加載的都是同一個類,若是不使用雙親委派模型,由各個類加載器自行去加載的話,系統中就會出現不止一個Object類,應用程序就會全亂了.

Class.forname()與ClassLoader.loadClass():

  Class.forname():是一個靜態方法,最經常使用的是Class.forname(String className);根據傳入的類的全限定名返回一個Class對象.該方法在將Class文件加載到內存的同時,會執行類的初始化.

  如: Class.forName("com.wang.HelloWorld");

  ClassLoader.loadClass():這是一個實例方法,須要一個ClassLoader對象來調用該方法,該方法將Class文件加載到內存時,並不會執行類的初始化,直到這個類第一次使用時才進行初始化.該方法由於須要獲得一個ClassLoader對象,因此能夠根據須要指定使用哪一個類加載器.

  如:ClassLoader cl=.......;cl.loadClass("com.wang.HelloWorld");

  


 

 注:整理這篇筆記,參考了<深刻理解java虛擬機>這本書,以及深刻探討 Java 類加載器這篇文章,想深刻了解能夠去看看.

相關文章
相關標籤/搜索