這篇文章主要介紹“java虛擬機(jī)的類加載機(jī)制介紹”,在日常操作中,相信很多人在java虛擬機(jī)的類加載機(jī)制介紹問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對(duì)大家解答”java虛擬機(jī)的類加載機(jī)制介紹”的疑惑有所幫助!接下來,請(qǐng)跟著小編一起來學(xué)習(xí)吧!
10年積累的成都做網(wǎng)站、網(wǎng)站設(shè)計(jì)經(jīng)驗(yàn),可以快速應(yīng)對(duì)客戶對(duì)網(wǎng)站的新想法和需求。提供各種問題對(duì)應(yīng)的解決方案。讓選擇我們的客戶得到更好、更有力的網(wǎng)絡(luò)服務(wù)。我雖然不認(rèn)識(shí)你,你也不認(rèn)識(shí)我。但先網(wǎng)站設(shè)計(jì)后付款的網(wǎng)站建設(shè)流程,更有南昌縣免費(fèi)網(wǎng)站建設(shè)讓你可以放心的選擇與我們合作。
類加載的時(shí)機(jī)
說到類加載的時(shí)機(jī),就不得不提類加載的生命周期了,加載——驗(yàn)證——準(zhǔn)備——解析——初始化——使用——卸載。七個(gè)階段。書中說解析和初始化的位置可以互換,是為了支持java語言的動(dòng)態(tài)綁定導(dǎo)致的。這點(diǎn)我暫時(shí)不知道,不做研究先。
不過我們一起來探討下類加載什么情況下會(huì)立即觸發(fā)初始化呢。
java虛擬機(jī)規(guī)范中明確規(guī)定了五種情況必須立即對(duì)類進(jìn)行初始化:
遇到new、getstatic、putstatic或者invokestatic 這4條字節(jié)碼指令時(shí),如果類沒有初始化,則會(huì)先觸發(fā)類進(jìn)行初始化。這4條指令對(duì)應(yīng)最常見的java代碼場景為:使用new關(guān)鍵字實(shí)例化對(duì)象的時(shí)候、讀取或者設(shè)置一個(gè)類的靜態(tài)變量(static final修飾的變量不算,因?yàn)樵诰幾g期就把該變量放入常量池中了)、調(diào)用一個(gè)類的靜態(tài)方法。
使用java.lang.reflect包的方法對(duì)類進(jìn)行放射調(diào)用時(shí),如果沒有初始化,會(huì)優(yōu)先初始化
初始化一個(gè)類的時(shí)候,如果發(fā)現(xiàn)其父類還沒有初始化,會(huì)優(yōu)先初始化其父類
當(dāng)虛擬機(jī)啟動(dòng)時(shí),用戶指定一個(gè)要執(zhí)行的主類(main()方法的類),會(huì)優(yōu)先初始化主類
如果java.lang.invoke.MethodHandle實(shí)例最后解析的結(jié)果REF_getstatic、REF_putstatic、ERF_invokestatic的方法句柄,且這個(gè)方法的句柄所對(duì)應(yīng)的類沒有初始化,則會(huì)優(yōu)先初始化。
以上的5中場景成為對(duì)一個(gè)類的主動(dòng)引用,除此之外,所以引用類的方式都不會(huì)觸發(fā)初始化,成為被動(dòng)引用。
下面看看代碼:
package chapter.Seven;
class SuperClass {
static{
System.out.println("SuperClass init");
}
public static int value=123;
public static final String NAME="QuellanAn";
}
class SubClass extends SuperClass {
static{
System.out.println("SubClass init");
}
}
public class NotInitialzation {
public static void main(String[] args){
System.out.println(SubClass.value);
}
}
main方法中調(diào)用“SubClass.value”,value是SuperClass的靜態(tài)變量。所以觸發(fā)了上面第一點(diǎn),所以SuperClass類會(huì)被初始化。雖然是通過SubClass調(diào)用的,但是SubClass類沒有觸發(fā)上面5點(diǎn)的任何一點(diǎn),所以不會(huì)進(jìn)行初始化。
結(jié)果:
SuperClass init123
在看一個(gè),就改下main()方法,其他都不不變。
public class NotInitialzation { public static void main(String[] args){ System.out.println(SubClass.NAME); }}
同樣根據(jù)上面第一點(diǎn),可以發(fā)現(xiàn),NAME是靜態(tài)常量,在編譯期(形成class文件的時(shí)候)就已經(jīng)存在常量池中了。所以SuperClass也是不會(huì)初始化的。
結(jié)果:
QuellanAn
上面的例子可以很好的看出什么時(shí)候?qū)@個(gè)類初始化了什么時(shí)候沒有。有且只有上面那5中情況才會(huì)對(duì)類型進(jìn)行初始化。
另外說一點(diǎn),當(dāng)一個(gè)類在初始化時(shí),要求其父類全部都已經(jīng)初始化了,但是在接口初始化時(shí),并不要求父接口全部初始化,只需要在正真用到哪個(gè)父接口就初始化那個(gè)。
類加載的過程
類加載的過程分為7個(gè)階段,但重要的前面加載——驗(yàn)證——準(zhǔn)備——解析——初始化前面5個(gè)。下面來依次的說明下。
加載
在加載階段,虛擬機(jī)需要完成以下3件事:
1. 通過一個(gè)類的全限定名來獲取定義此類的二進(jìn)制字節(jié)流。(全限定名:第一次聽到這個(gè)詞有點(diǎn)摸不著頭腦,網(wǎng)上查了一下,才知道是有點(diǎn)絕對(duì)路徑的意思,比如
Java類包的定名:com.linux.struct.sort.bubblesort,從最原始最上層的地方援引到具體的對(duì)象,這就是全限定名了)
2.將二進(jìn)制字節(jié)流中所代表的的靜態(tài)存儲(chǔ)結(jié)構(gòu)轉(zhuǎn)化為方法區(qū)運(yùn)行時(shí)結(jié)構(gòu)。
3.在內(nèi)存中生成一個(gè)代表這個(gè)類的java.lang.Class對(duì)象,作為方法區(qū)這個(gè)的類的各個(gè)數(shù)據(jù)的訪問入口。
驗(yàn)證
驗(yàn)證是連接階段的第一步,驗(yàn)證的目的主要是為了確保class文件轉(zhuǎn)換成的二進(jìn)制文件流中包含的信息符合當(dāng)前虛擬機(jī)的要求。并且不會(huì)危害到虛擬機(jī)的安全。
驗(yàn)證階段主要完成4個(gè)階段的驗(yàn)證工作:文件格式檢驗(yàn)、元數(shù)據(jù)校驗(yàn)、字節(jié)碼校驗(yàn)、符號(hào)引用校驗(yàn)。
準(zhǔn)備
準(zhǔn)備階段,是正式為類的變量分配內(nèi)存空間并設(shè)置類變量初始值的階段。這些變量所使用內(nèi)存都將在方法區(qū)中分配。需要注意的是:這個(gè)階段今次進(jìn)行內(nèi)存分配的進(jìn)包含類變量,而不包含實(shí)例變量。實(shí)例變量將在對(duì)象實(shí)例化時(shí)隨著對(duì)象一起分配到j(luò)ava堆中。并且這里說的初始值“通常情況”下是數(shù)據(jù)類型的零值,比如:
public static int value=123;
在value準(zhǔn)備階段過后的初始值為0,而不是123.value值在初始化的時(shí)候才會(huì)被賦值成123.因?yàn)榘裿alue賦值為123的putstatic指令在初始化后才執(zhí)行。
解析
解析階段是虛擬機(jī)將常量池內(nèi)的符號(hào)引用替換成直接引用的過程。
初始化
初始化階段,執(zhí)行類構(gòu)造器
到此,關(guān)于“java虛擬機(jī)的類加載機(jī)制介紹”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實(shí)踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識(shí),請(qǐng)繼續(xù)關(guān)注創(chuàng)新互聯(lián)網(wǎng)站,小編會(huì)繼續(xù)努力為大家?guī)砀鄬?shí)用的文章!