聊一聊Java的JVM類加載機制
Java虛擬機把描述類的數(shù)據(jù)從Class文件加載到內(nèi)存,并對數(shù)據(jù)進行校驗、轉(zhuǎn)換解析和初始化,最終形成可以被虛擬機直接使用的Java類型,這個過程被稱作虛擬機的類加載機制。當Java程序運行時,Java虛擬機會按需加載類,即在程序需要使用某個類時才會加載該類。
類的生命周期如下圖:

JVM的類加載機制包括加載、連接( 驗證、準備、解析)、初始化 3個階段。
加載(Loading)
加載(Loading) 階段主要是查找并加載字節(jié)碼文件,這個文件可以是來自本地文件系統(tǒng)、網(wǎng)絡、jar包等地方。加載后,生成一個對應的Class對象。
加載類時會做以下工作:
- 根據(jù)類的全限定名查找并讀取類的二進制數(shù)據(jù)。類的二進制數(shù)據(jù)可以來自文件、網(wǎng)絡、數(shù)據(jù)庫等各種數(shù)據(jù)源。
- 將類的二進制數(shù)據(jù)轉(zhuǎn)換成方法區(qū)內(nèi)部的數(shù)據(jù)結(jié)構(gòu)。在轉(zhuǎn)換的過程中,JVM會對類的二進制數(shù)據(jù)進行解析和校驗。
- 在方法區(qū)內(nèi)存儲該類的相關信息,包括類的名稱、修飾符、常量池、字段描述符、方法描述符、接口描述符、方法表等。
- 生成一個代表該類的Class對象,并將該對象存放在JVM的堆內(nèi)存中。Class對象包含了類的各種信息,可以用于創(chuàng)建類的實例、獲取類的方法和字段等操作。
需要注意的是,在加載類的過程中,JVM會遵循一定的雙親委派機制,即先委派給父類加載器嘗試加載,如果父類加載器無法加載,則由當前類加載器進行加載。這樣可以保證類的加載不會重復,避免出現(xiàn)類似的類被多次加載的情況。有關類加載器可以查看我之前的文章。
加載階段與連接階段的部分動作(如一部分字節(jié)碼文件格式驗證動作)是交叉進行的,加載階段尚未完成,連接階段可能已經(jīng)開始,但這些夾在加載階段之中進行的動作,仍然屬于連接階段的一部分,這兩個階段的開始時間仍然保持著固定的先后順序。
連接(Linking)
連接階段是Java虛擬機將類文件中的符號引用轉(zhuǎn)換為直接引用的過程,會對字節(jié)碼文件進行驗證、準備、解析。
- 驗證(Verification):在這個階段,JVM會對字節(jié)碼進行驗證,以確保其符合Java虛擬機規(guī)范,并且不會對虛擬機造成安全威脅。驗證的內(nèi)容包括靜態(tài)分析、字節(jié)碼驗證、符號引用驗證等。如果驗證失敗,JVM會拋出VerifyError異常。
- 準備(Preparation):在這個階段,JVM會為類的靜態(tài)變量分配內(nèi)存,并將其初始化為默認值(零值)。這個階段不會執(zhí)行任何Java代碼,只是為靜態(tài)變量分配內(nèi)存空間。例如將int類型的靜態(tài)變量賦值為0。
public static int staticValue = 123;
準備階段初始化只是將靜態(tài)變量初始化為默認值,比如上面這段代碼,不同數(shù)據(jù)類型都有其默認值,初始化是只是將staticValue賦予默認值0,也就是staticValue = 0,只有在初始化階段才會將staticValue賦值為123,也就是staticValue = 123。但是如果是staticValue是個常量public static final int staticValue = 123,準備階段才會將staticValue賦值為123。
- 解析(Resolution):在這個階段,JVM會將類、接口、字段和方法的符號引用解析為直接引用。符號引用是指用來描述某個類、字段或方法的名稱和類型的符號,而直接引用則是指直接指向內(nèi)存中的具體位置的引用。解析的過程包括將類或接口的符號引用解析為直接引用、將字段的符號引用解析為直接引用、將類中方法的符號引用解析為直接引用。
初始化(Initialization)
初始化階段是指在類被首次主動使用時執(zhí)行的階段,虛擬機會執(zhí)行類的初始化代碼,包括靜態(tài)變量的賦值和靜態(tài)代碼塊的執(zhí)行等操作。
初始化階段是類加載的最后一個階段,在該階段,JVM會執(zhí)行以下操作:
- 執(zhí)行靜態(tài)變量賦值操作:在這個階段,JVM會對所有的靜態(tài)變量進行初始化并賦值。這些靜態(tài)變量的值通常在類定義中已經(jīng)被明確定義了,JVM會根據(jù)定義進行相應的賦值操作。
- 執(zhí)行靜態(tài)代碼塊:如果類定義中包含有靜態(tài)代碼塊,那么在該階段JVM會執(zhí)行這些靜態(tài)代碼塊中的代碼。
- 調(diào)用類的初始化方法:在Java程序中,可以使用static關鍵字來定義一個類的靜態(tài)初始化方法(即static void methodName())。在該階段,JVM會調(diào)用這個類的靜態(tài)初始化方法。
類初始化的時機
類初始化時機包括以下四種情況:
- 創(chuàng)建該類的實例對象時,例如使用 new 關鍵字創(chuàng)建對象,類的初始化將被觸發(fā) 如果一個類是程序執(zhí)行的入口類(即包含 main() 方法的類),那么也會觸發(fā)該類的初始化操作。
- 子類初始化會觸發(fā)父類初始化:當一個子類被初始化時,其父類也會被初始化。這意味著,如果一個類沒有被使用,那么它的父類也不會被初始化。
- 當調(diào)用類的靜態(tài)方法(不包括final方法和private方法)或訪問類的靜態(tài)字段(不包括final字段)時,類的初始化將被觸發(fā)。
- 當使用反射API對類進行某些操作時(例如使用Class.forName()方法加載類、調(diào)用Class.newInstance()方法創(chuàng)建對象、調(diào)用Method.invoke()方法調(diào)用方法等),類的初始化將被觸發(fā)。
初始化是線程安全的JVM保證一個類的初始化只會由一個線程去執(zhí)行,其他線程需要等待該線程完成后才能訪問該類。
下面用一個簡單的Java代碼示例,展示JVM類加載機制中初始化階段的示例
public class MyClass {
// 靜態(tài)變量
public static String staticStr = "Hello, world!";
static {
System.out.println("MyClass is initialized.");
}
// 構(gòu)造方法
public MyClass() {
System.out.println("MyClass constructor is called.");
}
// 靜態(tài)方法
public static void staticMethod() {
System.out.println("MyClass staticMethod is called.");
}
}
在上述代碼中,類MyClass包含一個靜態(tài)變量staticStr、一個靜態(tài)代碼塊和一個構(gòu)造方法,以及一個靜態(tài)方法staticMethod。當程序首次使用MyClass類時,JVM將會觸發(fā)MyClass類的初始化階段??梢酝ㄟ^下面的代碼來測試類的初始化:
public class Test {
public static void main(String[] args) {
System.out.println(MyClass.staticStr); // 調(diào)用靜態(tài)變量,觸發(fā)類初始化
MyClass.staticMethod(); // 調(diào)用靜態(tài)方法,觸發(fā)類初始化
MyClass obj = new MyClass(); // 創(chuàng)建對象,觸發(fā)類初始化
}
}
在上面的代碼中,首先輸出了MyClass類的靜態(tài)變量staticStr,此時會觸發(fā)MyClass類的初始化;然后調(diào)用了靜態(tài)方法staticMethod,同樣會觸發(fā)MyClass類的初始化;最后創(chuàng)建了一個MyClass對象,也會觸發(fā)MyClass類的初始化。運行上述代碼,可以看到以下輸出:
MyClass is initialized. Hello, world! MyClass staticMethod is called. MyClass constructor is called.
輸出結(jié)果表明,MyClass類的初始化確實在首次使用該類時被觸發(fā),包括靜態(tài)變量、靜態(tài)代碼塊、靜態(tài)方法和構(gòu)造方法都被執(zhí)行了。
此外,如果一個類是另一個類的子類,那么在使用子類時,父類也會被初始化。例如:
public class MyBaseClass {
static {
System.out.println("MyBaseClass is initialized.");
}
}
public class MySubClass extends MyBaseClass {
static {
System.out.println("MySubClass is initialized.");
}
}
public class Test {
public static void main(String[] args) {
MySubClass obj = new MySubClass(); // 創(chuàng)建子類對象,觸發(fā)父類和子類初始化
}
}
在上述代碼中,當創(chuàng)建MySubClass類的對象時,將會觸發(fā)MyBaseClass和MySubClass類的初始化。運行上述代碼,可以看到以下輸出:
MyBaseClass is initialized. MySubClass is initialized.
總結(jié)
JVM的類加載機制采用了延遲加載的策略,即在需要使用類時才加載該類,這種方式可以提高程序的啟動速度,也避免了不必要的資源浪費。同時,JVM還提供了多個類加載器,可以通過自定義類加載器實現(xiàn)特定的加載策略,例如動態(tài)加載、遠程加載等,從而滿足不同的應用需求。
到此這篇關于聊一聊Java的JVM類加載機制的文章就介紹到這了,更多相關JVM類加載機制內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
mybatis整合springboot報BindingException:Invalid?bound?stateme
這篇文章主要給大家介紹了關于mybatis整合springboot報BindingException:Invalid?bound?statement?(not?found)異常的解決辦法,這個錯誤通常是由于Mapper文件中的statement?id與Java代碼中的方法名不一致導致的,需要的朋友可以參考下2024-01-01
spring-mvc/springboot使用MockMvc對controller進行測試
這篇文章主要介紹了spring-mvc/springboot使用MockMvc對controller進行測試,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-11-11
springboot發(fā)布dubbo服務注冊到nacos實現(xiàn)方式
這篇文章主要介紹了springboot發(fā)布dubbo服務注冊到nacos實現(xiàn)方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-09-09
Mybatis Mapper XML文件-插入,更新,刪除詳解(insert, updat
這篇文章主要介紹了MyBatis的Mapper XML文件中用于插入、更新和刪除數(shù)據(jù)的語句,包括這些語句的屬性和子元素的使用方法2025-02-02
java如何利用poi解析doc和docx中的數(shù)據(jù)
這篇文章主要給大家介紹了關于java如何利用poi解析doc和docx中數(shù)據(jù)的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2021-04-04
java并發(fā)編程synchronized底層實現(xiàn)原理
這篇文章主要介紹了java并發(fā)編程synchronized底層實現(xiàn)原理2022-02-02

