Java:類加載過(guò)程
1.加載――3件事
- 1.通過(guò)全類名獲取定義此類的二進(jìn)制字節(jié)流(eg:從jar、war中獲取);
- 2.將字節(jié)流所代表的靜態(tài)存儲(chǔ)結(jié)構(gòu)轉(zhuǎn)換為方法區(qū)的運(yùn)行時(shí)數(shù)據(jù)結(jié)構(gòu);
- 3.在內(nèi)存中生成一個(gè)代表該類的Class對(duì)象,作為方法區(qū)這些數(shù)據(jù)的訪問(wèn)入口。
2.連接
加載階段和連接階段的部分內(nèi)容是交叉進(jìn)行的,加載尚未結(jié)束,連接階段可能就開(kāi)始運(yùn)行了。
2.1.驗(yàn)證
2.2.準(zhǔn)備
準(zhǔn)備階段:正式為類變量分配內(nèi)存并設(shè)置類變量初始值的階段,這些內(nèi)存在方法區(qū)分配。注意:
1.這時(shí)候進(jìn)行內(nèi)存分配的僅包括類變量(static),而不包括實(shí)例變量,實(shí)例變量會(huì)在對(duì)象實(shí)例化時(shí)隨著對(duì)象一塊分配在 Java 堆中;
2.除了含有final修飾的變量外,其他都賦值0,null,false這種的。
如下例子, value 的值就被復(fù)制為 111,而不是0.
public static final int value=111
2.3.解析
解析階段是虛擬機(jī)將常量池中的符號(hào)引用替換為直接引用的過(guò)程。
- 也就是得到類或者字段、方法在內(nèi)存中的指針或者偏移量。
什么是符號(hào)引用?
* 比如org.simple.People類引用了org.simple.Language類;
* 在編譯時(shí)People類并不知道Language類的實(shí)際內(nèi)存地址,因此只能使用符號(hào)org.simple.Language
3.初始化
初始化是類加載的最后一步,也是真正執(zhí)行類中定義的 Java 程序代碼(字節(jié)碼),初始化階段是執(zhí)行類構(gòu)造器< clinit >()方法的過(guò)程。
對(duì)于< clinit >() 方法的調(diào)用,虛擬機(jī)會(huì)自己確保其在多線程環(huán)境中的安全性。因?yàn)?< clinit >() 方法是帶鎖線程安全,所以在多線程環(huán)境下進(jìn)行類初始化的話可能會(huì)引起死鎖,并且這種死鎖很難被發(fā)現(xiàn)。
對(duì)于初始化階段,虛擬機(jī)嚴(yán)格規(guī)范了有且只有5中情況下,必須對(duì)類進(jìn)行初始化:
- 當(dāng)遇到 new 、 getstatic、putstatic或invokestatic 這4條直接碼指令時(shí),比如 new 一個(gè)類,讀取一個(gè)靜態(tài)字段(未被 final 修飾)、或調(diào)用一個(gè)類的靜態(tài)方法時(shí)。
- 使用 java.lang.reflect 包的方法對(duì)類進(jìn)行反射調(diào)用時(shí) ,如果類沒(méi)初始化,需要觸發(fā)其初始化.
- 初始化一個(gè)類,如果其父類還未初始化,則先觸發(fā)該父類的初始化。
- 當(dāng)虛擬機(jī)啟動(dòng)時(shí),用戶需要定義一個(gè)要執(zhí)行的主類 (包含 main 方法的那個(gè)類),虛擬機(jī)會(huì)先初始化這個(gè)類。
-
當(dāng)使用 JDK1.7 的動(dòng)態(tài)動(dòng)態(tài)語(yǔ)言時(shí),如果一個(gè) MethodHandle 實(shí)例的最后解析結(jié)構(gòu)為
REF_getStatic
、REF_putStatic
、REF_invokeStatic
、的方法句柄,并且這個(gè)句柄沒(méi)有初始化,則需要先觸發(fā)器初始化。
總結(jié)
本篇文章就到這里了,希望能夠給你帶來(lái)幫助,也希望您能夠多多關(guān)注服務(wù)器之家的更多內(nèi)容!
原文鏈接:https://blog.csdn.net/m0_38057941/article/details/120106993