Java 虛擬機(jī)中的類加載即從 class 文件到內(nèi)存中的類,按先后順序需要經(jīng)過加載、鏈接以及初始化三大步驟。
虛擬機(jī)的加載對象是什么?
上文中說過Java中有兩種類型:基本類型和引用類型,而基本類型是由虛擬機(jī)預(yù)先定義好的,引用類型中的泛型參數(shù)又會在編譯過程中被擦除,所以加載的對象就剩下類、接口和數(shù)組類。
在類、接口和數(shù)組類中,數(shù)組類是由 Java 虛擬機(jī)直接生成的,其他兩種則有對應(yīng)的字節(jié)流。無論是直接生成的數(shù)組類,還是加載的類,Java 虛擬機(jī)都需要對其進(jìn)行鏈接和初始化。接下來,就詳細(xì)介紹一下每個(gè)步驟具體都在干些什么。
虛擬機(jī)的加載流程是什么?
1.加載
是指查找字節(jié)流,并且據(jù)此創(chuàng)建類的過程。上面提過數(shù)組類是由Java虛擬機(jī)直接生成的,所以加載過程針對的是生成字節(jié)流的類與接口。如何找到這些字節(jié)流,則需要虛擬機(jī)借助類加載器。
啟動類加載器是由 C++ 實(shí)現(xiàn)的,沒有對應(yīng)的 Java 對象,因此在 Java 中只能用 null 來指代。在 Java 9 之前,啟動類加載器負(fù)責(zé)加載最為基礎(chǔ)、最為重要的類,比如存放在 JRE 的 lib 目錄下 jar 包中的類(以及由虛擬機(jī)參數(shù) -Xbootclasspath 指定的類)。除了啟動類加載器之外,另外兩個(gè)重要的類加載器是擴(kuò)展類加載器(extension class loader)和應(yīng)用類加載器(application class loader),均由 Java 核心類庫提供。故除了啟動類加載器之外,其他的類加載器都是java.lang.ClassLoader 的子類,因此有對應(yīng)的 Java 對象。
擴(kuò)展類加載器的父類加載器是啟動類加載器。它負(fù)責(zé)加載相對次要、但又通用的類,比如存放在 JRE 的 lib/ext 目錄下 jar 包中的類(以及由系統(tǒng)變量 java.ext.dirs 指定的類)。
應(yīng)用類加載器的父類加載器則是擴(kuò)展類加載器。它負(fù)責(zé)加載應(yīng)用程序路徑下的類。(這里的應(yīng)用程序路徑,便是指虛擬機(jī)參數(shù) -cp/-classpath、系統(tǒng)變量 java.class.path 或環(huán)境變量 CLASSPATH 所指定的路徑。)默認(rèn)情況下,應(yīng)用程序中包含的類便是由應(yīng)用類加載器加載的。
Java 9 引入了模塊系統(tǒng),并且略微更改了上述的類加載器1。擴(kuò)展類加載器被改名為平臺類加載器(platform class loader)。Java SE 中除了少數(shù)幾個(gè)關(guān)鍵模塊,比如說 java.base 是由啟動類加載器加載之外,其他的模塊均由平臺類加載器所加載。當(dāng)然還可以自定義類加載器哦。
除了加載功能之外,類加載器還提供了命名空間的作用,在 Java 虛擬機(jī)中,類的唯一性是由類加載器實(shí)例以及類的全名一同確定的。即便是同一串字節(jié)流,經(jīng)由不同的類加載器加載,也會得到兩個(gè)不同的類。在大型應(yīng)用中,我們往往借助這一特性,來運(yùn)行同一個(gè)類的不同版本。
2.鏈接
是指將創(chuàng)建成的類合并至 Java 虛擬機(jī)中,使之能夠執(zhí)行的過程。它可分為驗(yàn)證、準(zhǔn)備以及解析三個(gè)階段。
- 驗(yàn)證階段:確保被加載類能夠滿足 Java 虛擬機(jī)的約束條件。
- 準(zhǔn)備階段:為被加載類的靜態(tài)字段分配內(nèi)存,構(gòu)造其他跟類層次相關(guān)的數(shù)據(jù)結(jié)構(gòu)。
- 解析階段:將符號引用解析成為實(shí)際引用(Java 虛擬機(jī)規(guī)范并沒有要求在鏈接過程中完成解析。它僅規(guī)定了:如果某些字節(jié)碼使用了符號引用,那么在執(zhí)行這些字節(jié)碼之前,需要完成對這些符號引用的解析)。
符號引用則是在 class 文件被加載至 Java 虛擬機(jī)之前,類無法知道其他類及其方法、字段所對應(yīng)的具體地址,甚至不知道自己方法、字段的地址。每當(dāng)需要引用這些成員時(shí),Java 編譯器會生成一個(gè)符號引用。在運(yùn)行階段,這個(gè)符號引用一般都能夠無歧義地定位到具體目標(biāo)上。
3.初始化
初始化即給常量賦值以及執(zhí)行 < clinit > 方法的過程,完成之后,類才正式成為可執(zhí)行的狀態(tài)。
類初始化觸發(fā)條件
- 當(dāng)虛擬機(jī)啟動時(shí),初始化用戶指定的主類;
- 當(dāng)遇到用以新建目標(biāo)類實(shí)例的 new 指令時(shí),初始化 new 指令的目標(biāo)類;
- 當(dāng)遇到調(diào)用靜態(tài)方法的指令時(shí),初始化該靜態(tài)方法所在的類;
- 當(dāng)遇到訪問靜態(tài)字段的指令時(shí),初始化該靜態(tài)字段所在的類;
- 子類的初始化會觸發(fā)父類的初始化;
- 如果一個(gè)接口定義了 default 方法,那么直接實(shí)現(xiàn)或者間接實(shí)現(xiàn)該接口的類的初始化,會觸發(fā)該接口的初始化;
- 使用反射 API 對某個(gè)類進(jìn)行反射調(diào)用時(shí),初始化這個(gè)類;
- 當(dāng)初次調(diào)用 MethodHandle 實(shí)例時(shí),初始化該 MethodHandle 指向的方法所在的類。
總結(jié)
虛擬機(jī)加載Java類是Java 虛擬機(jī)將字節(jié)流轉(zhuǎn)化為 Java 類的過程。這個(gè)過程可分為加載、鏈接以及初始化三大步驟。
-
加載:是指查找字節(jié)流,并且據(jù)此創(chuàng)建類的過程。加載需要借助類加載器,在 Java 虛擬機(jī)中,類加載器使用了雙親委派模型,即接收到加載請求時(shí),會先將請求轉(zhuǎn)發(fā)給父類加載器。 -
鏈接:是指將創(chuàng)建成的類合并至 Java 虛擬機(jī)中,使之能夠執(zhí)行的過程。鏈接還分驗(yàn)證、準(zhǔn)備和解析三個(gè)階段。其中,解析階段為非必須的。 -
初始化:是為標(biāo)記為常量值的字段賦值,以及執(zhí)行 < clinit > 方法的過程。類的初始化僅會被執(zhí)行一次,這個(gè)特性被用來實(shí)現(xiàn)單例的延遲初始化。