jvm虛擬機(jī)類加載機(jī)制詳解
1 概述
? Java虛擬機(jī)把描述類的數(shù)據(jù)從Class文件加載到內(nèi)存, 并對數(shù)據(jù)進(jìn)行校驗(yàn)、轉(zhuǎn)化解析和初始化,最終形成可以被虛擬機(jī)直接使用的Java類型,這個(gè)過程稱為虛擬機(jī)的類加載機(jī)制。在Java語言中,類型的加載、連接和初始化都是在程序運(yùn)行期間完成的。
2 類的加載時(shí)機(jī)
? 一個(gè)類型從被加載到虛擬機(jī)內(nèi)存中開始,到卸載出內(nèi)存為止,它的整個(gè)生命周期將會(huì)經(jīng)歷加載、驗(yàn)證、準(zhǔn)備、解析、初始化、使用和卸載七個(gè)階段,其中驗(yàn)證、準(zhǔn)備和解析三個(gè)部分統(tǒng)稱為連接。發(fā)生順序如下:

? 《Java虛擬機(jī)規(guī)范》嚴(yán)格規(guī)定有且只有六種情況必須立即對類進(jìn)行“初始化”(加載、驗(yàn)證和準(zhǔn)備自然需要在此之前開始):
- 遇到new、getstatic、putstatic或invokestatic這四條字節(jié)碼指令時(shí),如果類型沒有進(jìn)行初始化,則需要先觸發(fā)其初始化階段。能夠生成這四條指令的典型Java代碼場景有:
- 使用new關(guān)鍵字實(shí)例化對象的時(shí)候
- 讀取或設(shè)置一個(gè)類型的靜態(tài)字段的時(shí)候
- 調(diào)用一個(gè)類型的靜態(tài)方法的時(shí)候
- 使用java.lang.reflect包的方法對類型進(jìn)行反射調(diào)用的時(shí)候,如果還沒被初始化,則需要先觸發(fā)其初始化。
- 當(dāng)初始化類的時(shí)候,如果發(fā)現(xiàn)其父類還沒進(jìn)行過初始化,則需要先觸發(fā)其父類的初始化。
- 當(dāng)虛擬機(jī)啟動(dòng)時(shí),用戶需要指定一個(gè)要執(zhí)行的主類。虛擬機(jī)會(huì)先初始化這個(gè)主類。
- jdk1.7新加入的動(dòng)態(tài)語言支持時(shí)。
- 當(dāng)一個(gè)接口中定義了jdk8新加入的默認(rèn)方法時(shí)。
? 這六種場景的行為稱為對一個(gè)類型進(jìn)行主動(dòng)引用。除此之外,所有引用類型的方式都不會(huì)觸發(fā)初始化,稱為被動(dòng)引用。下面舉出被動(dòng)引用的例子
示例1:通過子類引用父類的靜態(tài)字段,不會(huì)導(dǎo)致子類初始化
public class NotInitialization {
public static void main(String[] args) {
System.out.println(SubClass.value); // 只會(huì)輸出“SuperClass init”,而不會(huì)輸出“subclass init”
SuperClass[] superClasses = new SuperClass[10]; // 不會(huì)輸出“SuperClass init”
}
}
class SuperClass {
static {
System.out.println("SuperClass init");
}
public static int value = 123;
}
class SubClass extends SuperClass {
static {
System.out.println("SubClass init!");
}
}
示例2:常量在編譯階段會(huì)存入調(diào)用類的常量池中,本質(zhì)上沒有直接引用到定義常量的類,因此不會(huì)觸發(fā)定義常量的初始化
public class NotInitialization {
public static void main(String[] args) {
System.out.println(ConstClass.HELLOWORLD); // 不會(huì)輸出“ConstClass init!”,因?yàn)槌A恐苯哟鎯?chǔ)到常量池中。
}
}
class ConstClass {
static {
System.out.println("ConstClass init!");
}
public static final String HELLOWORLD = "hello world";
}
class Test {
static {
i = 0;
}
static int i = 1;
}
3 類的加載過程
3.1 加載
? 在加載階段,虛擬機(jī)需要完成以下三件事情:
通過一個(gè)類的全限定名來獲取定義此類的二進(jìn)制字節(jié)流。將這個(gè)字節(jié)流所代表的靜態(tài)存儲(chǔ)結(jié)構(gòu)轉(zhuǎn)化為方法區(qū)的運(yùn)行時(shí)數(shù)據(jù)結(jié)構(gòu)。在內(nèi)存中生成一個(gè)代表這個(gè)類的java.lang.Class對象,作為方法區(qū)這個(gè)類的的各種數(shù)據(jù)的訪問入口。
3.2 驗(yàn)證
? 驗(yàn)證是連接階段的第一步,這一階段的目的是確保Class文件的字節(jié)流中包含的信息符合《Java虛擬機(jī)規(guī)范》的約束要求,保證這些信息不會(huì)危害虛擬機(jī)自身。
3.3 準(zhǔn)備
? 準(zhǔn)備階段是正式為類中定義的變量(即靜態(tài)變量,被static修飾的變量)分配內(nèi)存并設(shè)置類變量初始值的階段。
3.4 解析
? 解析階段是將常量池內(nèi)的符號引用替換為直接引用的過程。
3.5 初始化
? 初始化是類加載過程的最后一個(gè)階段,直到初始化階段,Java虛擬機(jī)才真正開始執(zhí)行類中編寫的Java程序代碼,將主導(dǎo)權(quán)移交給應(yīng)用程序。初始化階段就是執(zhí)行類構(gòu)造器的()方法的過程。
- ()方法與類的構(gòu)造函數(shù)不同,它不需要顯示地調(diào)用父類構(gòu)造器,Java虛擬機(jī)會(huì)保證在子類的()方法執(zhí)行前,父類的()方法已經(jīng)執(zhí)行完畢。因此在Java虛擬機(jī)中第一個(gè)被執(zhí)行的()方法肯定是Object。
- 由于父類的()方法先執(zhí)行,也就意味著父類中定義的靜態(tài)語句塊要優(yōu)先于子類的變量賦值操作,如下代碼:字段值將會(huì)是2而不是1。
public static void main(String[] args) {
System.out.println(Sub.B);
}
static class Parent {
public static int A = 1;
static {
A = 2;
}
}
static class Sub extends Parent {
public static int B = A;
}
4 類加載器
? 類加載器用于實(shí)現(xiàn)類的加載動(dòng)作,JVM中內(nèi)置了三個(gè)重要的ClassLoader,除了BootstrapClassLoader其他類加載器均由Java實(shí)現(xiàn)且全部繼承自java.lang.ClassLoader:
- BootstrapClassLoader(啟動(dòng)類加載器):最頂層的加載類,負(fù)責(zé)加載
%JAVA_HOME%/lib目錄下的jar包和類或者被-X:bootclasspathc參數(shù)指定的路徑下的所有類。 - Extension Class Loader(擴(kuò)展類加載器):主要負(fù)責(zé)加載<JAVA_HOME>\lib\ext目錄中的所有類庫。
- AppClassLoader(應(yīng)用程序類加載器):面向我們用戶的加載器,負(fù)責(zé)加載當(dāng)前應(yīng)用的classpath中的所有jar包和類。
4.1 雙親委派模型

? 如上圖:雙親委派模型的工作流程是如果一個(gè)類加載器收到了類加載的請求,它首先不會(huì)自己去嘗試加載這個(gè)類,而是把這個(gè)請求委派給父類加載器去完成,每一個(gè)層次的類加載器都是如此,因此所有的加載請求最終都應(yīng)該傳送到最頂層的啟動(dòng)類加載器中,只有當(dāng)父加載器反饋?zhàn)约簾o法完成這個(gè)加載請求時(shí),子加載器才會(huì)嘗試自己去完成加載。
好處
Java中的類隨著它的類加載器一起具備了一種帶有優(yōu)先級的層次關(guān)系。
實(shí)現(xiàn)
private final ClassLoader parent;
protected Class<?> loadClass(String name, boolean resolve)
throws ClassNotFoundException
{
synchronized (getClassLoadingLock(name)) {
// 首先,檢查請求的類是否已經(jīng)被加載過
Class<?> c = findLoadedClass(name);
if (c == null) {
long t0 = System.nanoTime();
try {
if (parent != null) {//父加載器不為空,調(diào)用父加載器loadClass()方法處理
c = parent.loadClass(name, false);
} else {//父加載器為空,使用啟動(dòng)類加載器 BootstrapClassLoader 加載
c = findBootstrapClassOrNull(name);
}
} catch (ClassNotFoundException e) {
//拋出異常說明父類加載器無法完成加載請求
}
if (c == null) {
long t1 = System.nanoTime();
//自己嘗試加載
c = findClass(name);
// this is the defining class loader; record the stats
sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
sun.misc.PerfCounter.getFindClasses().increment();
}
}
if (resolve) {
resolveClass(c);
}
return c;
}
}
4.2 破壞雙親委派模型
? 雙親委派模型主要出現(xiàn)過3次較大規(guī)模的“被破壞”的情況。具體想了解的詳看《深入理解Java虛擬機(jī)》。
到此這篇關(guān)于 jvm虛擬機(jī)類加載機(jī)制詳解的文章就介紹到這了,更多相關(guān) jvm虛擬機(jī)類加載內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring?Boot項(xiàng)目傳參校驗(yàn)的最佳實(shí)踐指南
有參數(shù)傳遞的地方都少不了參數(shù)校驗(yàn),在web開發(fā)中前端的參數(shù)校驗(yàn)是為了用戶體驗(yàn),后端的參數(shù)校驗(yàn)是為了安全,下面這篇文章主要給大家介紹了關(guān)于Spring?Boot項(xiàng)目傳參校驗(yàn)的最佳實(shí)踐,需要的朋友可以參考下2022-04-04
JavaFX Application應(yīng)用實(shí)例
下面小編就為大家?guī)硪黄狫avaFX Application應(yīng)用實(shí)例。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2016-10-10
在SpringBoot中配置Thymeleaf的模板路徑方式
這篇文章主要介紹了在SpringBoot中配置Thymeleaf的模板路徑方式,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-08-08
SpringBoot微服務(wù)注冊分布式Consul的詳細(xì)過程
這篇文章主要介紹了SpringBoot(微服務(wù))注冊分布式Consul,Spring Boot應(yīng)用可以通過向Consul注冊自身來實(shí)現(xiàn)服務(wù)發(fā)現(xiàn)和治理,使得其他服務(wù)可以在Consul中發(fā)現(xiàn)并調(diào)用它,需要的朋友可以參考下2023-04-04

