JVM學(xué)習(xí)筆記(三)——虛擬機(jī)類加載機(jī)制

在介紹完class文件格式后瓤漏,我們來看下虛擬機(jī)是如何把一個(gè)由class文件描述的類加載到內(nèi)存中的。具體來說java中類的加載涉及7個(gè)階段:加載蝶俱、校驗(yàn)、準(zhǔn)備饥漫、解析榨呆、初始化、使用积蜻、卸載。

1.加載時(shí)機(jī)

并不是所有的類在程序啟動(dòng)時(shí)即被加載竿拆,為提升效率,虛擬機(jī)通常秉承的是按需加載的原則宾尚,即需要使用到相應(yīng)的類時(shí)才加載對(duì)應(yīng)的類丙笋。具體包括如下幾個(gè)加載時(shí)機(jī):

  • 遇到new、getstatic央勒、putstatic、invokestatic這4條指令時(shí)稳吮,如果對(duì)應(yīng)的類沒有被加載,虛擬機(jī)會(huì)首先加載對(duì)應(yīng)的類灶似。這4條指令對(duì)應(yīng)的場(chǎng)景是:
    • 創(chuàng)建一個(gè)實(shí)例對(duì)象
    • 訪問一個(gè)類的靜態(tài)變量(注意:不包括被final修飾,在編譯時(shí)已被放入常量池的變量)
    • 執(zhí)行一個(gè)類的靜態(tài)方法
  • 使用java.lang.reflect包的方法對(duì)類進(jìn)行反射調(diào)用時(shí)酪惭,如果相應(yīng)類未被加載,則虛擬機(jī)會(huì)加載該類
  • 初始化子類時(shí)如果其父類尚未被加載春感,虛擬機(jī)會(huì)先加載其父類
  • 虛擬機(jī)啟動(dòng)時(shí),包含main方法的類會(huì)被加載
  • 使用JDK 1.7動(dòng)態(tài)語(yǔ)言支持時(shí)嫩实,某些場(chǎng)景會(huì)觸發(fā)類加載
  1. 加載

加載是整個(gè)類加載的一個(gè)過程窥岩,具體來說加載階段一共做了三項(xiàng)工作:

  • 通過一個(gè)類的全限定名來獲取定義此類的二進(jìn)制字節(jié)流
  • 將字節(jié)流中的靜態(tài)存儲(chǔ)結(jié)構(gòu)轉(zhuǎn)化為運(yùn)行中的實(shí)際數(shù)據(jù)結(jié)構(gòu)并存儲(chǔ)在方法區(qū)中
  • 為該類生成一個(gè)java.lang.Class對(duì)象,作為方法區(qū)這個(gè)類的各種數(shù)據(jù)的訪問入口

3.驗(yàn)證

驗(yàn)證階段的目的就是保證Class文件的字節(jié)流中包含的信息都符合當(dāng)前虛擬機(jī)的要求颂翼,不會(huì)危害虛擬機(jī)本身的安全。具體來說驗(yàn)證階段的工作主要分為以下幾部分:

3.1 文件格式驗(yàn)證

  • 是否已0xCAFEBABE開頭
  • 主次版本是否在當(dāng)前虛擬機(jī)可以處理的范圍內(nèi)
  • 常量池中的數(shù)據(jù)是否有不被支持的類型
  • 指向常量的各種索引值是否有指向不存在常量或不符要求的常量
  • …...

3.2 元數(shù)據(jù)驗(yàn)證

  • 當(dāng)前加載類是否有父類
  • 是否繼承了不被允許繼承的類(final類)
  • 如果不是抽象類球及,是否實(shí)現(xiàn)了父類中所有要求實(shí)現(xiàn)的方法
  • …...

3.3 字節(jié)碼驗(yàn)證

字節(jié)碼驗(yàn)證是整個(gè)驗(yàn)證過程中最為復(fù)雜的一步集歇,主要的目的是通過分析數(shù)據(jù)流和控制流,確定語(yǔ)義是合法的诲宇、符合邏輯的,例如:

  • 保證跳轉(zhuǎn)指令不會(huì)跳轉(zhuǎn)到方法體以外的字節(jié)碼指令上
  • 保證方法體內(nèi)的類型轉(zhuǎn)換是有效的
  • …...

3.4 符號(hào)引用驗(yàn)證

  • 符號(hào)引用中通過字符串描述的全限定名是否能夠找到對(duì)應(yīng)的類
  • 符號(hào)引用中的類姑蓝、字段、方法的訪問性(private纺荧、protected旭愧、public宙暇、default)是否可被當(dāng)前的類訪問

4 準(zhǔn)備

正式為類變量分配內(nèi)存并設(shè)置其初始值,這些變量所使用的內(nèi)存都將在方法區(qū)進(jìn)行分配占贫。

5 解析

解析是虛擬機(jī)將class文件中常量池中的符號(hào)引用解析為直接應(yīng)用的過程。

  • 符號(hào)引用:符號(hào)引用以一組符號(hào)來描述所引用的目標(biāo)瞳收,符號(hào)可以是任意形式的字面量,只要使用時(shí)能無歧義的定位到目標(biāo)即可螟深,與虛擬機(jī)的內(nèi)存布局無關(guān)。
  • 直接引用:直接引用可以直接訪問到存在于內(nèi)存中的目標(biāo)凡蜻,可以是一個(gè)直接指針也可以是一個(gè)句柄。

解析過程主要涉及以下幾個(gè)步驟:

  • 類或接口的解析
  • 字段解析
  • 類方法解析
  • 接口方法解析
  • 方法類型解析
  • 方法句柄解析
  • 調(diào)用點(diǎn)限定符解析

6 初始化

初始化就是執(zhí)行類構(gòu)造器方法<cinit>()的過程咽瓷,<cinit>()方法是由編譯器自動(dòng)收集的所有類變量的賦值動(dòng)作以及靜態(tài)語(yǔ)塊合并生成的舰讹。

7 類加載器

上述的類加載過程都是由java虛擬機(jī)的類加載器完成的闪朱。對(duì)于任意一個(gè)類,都需要有加載它的類加載器和這個(gè)類本身一同確立其在java虛擬機(jī)中的唯一性奋姿,每一個(gè)類加載器都擁有一個(gè)獨(dú)立的類命名空間。事實(shí)上Java程序在運(yùn)行時(shí)存在不止一種類加載器萍悴,絕大部分Java程序都會(huì)使用到以下三種類加載器:

  • 啟動(dòng)類加載器:用于加載<JAVA_HOME>/lib路徑下的類
  • 擴(kuò)展加載器:用于加載<JAVA_HOME>/lib/ext路徑下的類
  • 應(yīng)用程序類加載器:復(fù)雜加載用戶應(yīng)用程序路徑上的類

如果有需要寓免,開發(fā)人員還可以加入自定義的類加載器。既然存在如此多的類加載器袜香,那么當(dāng)一個(gè)類需要加載時(shí),具體是由那個(gè)類進(jìn)行加載呢蜈首?由于所有的類加載器都遵守“雙親委派模型”,所以虛擬機(jī)在運(yùn)行期間可以保證一個(gè)類只會(huì)被加載一次吆寨。


這里寫圖片描述

雙親委派模型的工作過程:如果一個(gè)類加載器收到了類加載的請(qǐng)求踩寇,它會(huì)把這個(gè)請(qǐng)求交給自己的父類加載器去完成,父類加載器也會(huì)繼續(xù)上自己的父類加載器發(fā)送請(qǐng)求盒延,依次類推缩擂。如果父類已經(jīng)加載過該類添寺,則當(dāng)前加載器會(huì)直接返回已加載的類,只有當(dāng)父類沒有加載過該類時(shí)博脑,當(dāng)前類加載器才會(huì)真正去加載該類票罐。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市该押,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌烟具,老刑警劉巖奠蹬,帶你破解...
    沈念sama閱讀 219,110評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異囤躁,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)言蛇,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,443評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門严沥,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人消玄,你說我怎么就攤上這事∈馨猓” “怎么了?”我有些...
    開封第一講書人閱讀 165,474評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵勘高,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我蕊蝗,道長(zhǎng),這世上最難降的妖魔是什么蓬戚? 我笑而不...
    開封第一講書人閱讀 58,881評(píng)論 1 295
  • 正文 為了忘掉前任宾抓,我火速辦了婚禮,結(jié)果婚禮上石洗,老公的妹妹穿的比我還像新娘。我一直安慰自己缕棵,他們只是感情好涉兽,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,902評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著花椭,像睡著了一般房午。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上郭厌,一...
    開封第一講書人閱讀 51,698評(píng)論 1 305
  • 那天折柠,我揣著相機(jī)與錄音,去河邊找鬼扇售。 笑死,一個(gè)胖子當(dāng)著我的面吹牛承冰,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播寂屏,決...
    沈念sama閱讀 40,418評(píng)論 3 419
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼迁霎!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起秘豹,我...
    開封第一講書人閱讀 39,332評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤芝此,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后婚苹,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,796評(píng)論 1 316
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡怎炊,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,968評(píng)論 3 337
  • 正文 我和宋清朗相戀三年廓译,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片非区。...
    茶點(diǎn)故事閱讀 40,110評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡征绸,死狀恐怖久橙,靈堂內(nèi)的尸體忽然破棺而出管怠,到底是詐尸還是另有隱情,我是刑警寧澤渤弛,帶...
    沈念sama閱讀 35,792評(píng)論 5 346
  • 正文 年R本政府宣布她肯,位于F島的核電站佳头,受9級(jí)特大地震影響晴氨,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜瑞筐,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,455評(píng)論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望块蚌。 院中可真熱鬧,春花似錦峭范、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,003評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)舶掖。三九已至,卻和暖如春尔店,著一層夾襖步出監(jiān)牢的瞬間眨攘,已是汗流浹背嚣州。 一陣腳步聲響...
    開封第一講書人閱讀 33,130評(píng)論 1 272
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留情竹,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,348評(píng)論 3 373
  • 正文 我出身青樓鲤妥,卻偏偏與公主長(zhǎng)得像拱雏,于是被迫代替她去往敵國(guó)和親底扳。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,047評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容