JVM類加載器,類加載機制詳解斗锭,看這一篇就夠了

前言

今天我們來講講 jvm 里 類加載 的過程地淀,我們寫了那么多類,卻不知道類的加載過程岖是,豈不是很尷尬 帮毁。

jvm 的啟動是通過引導類加載器(bootstrap class loader)創(chuàng)建一個初始類(initial class)來完成的,這個類是由 jvm 的具體實現(xiàn)指定的豺撑。[來自官方規(guī)范]

jvm組成結(jié)構(gòu)之一就是 類裝載器子系統(tǒng) 烈疚,我們今天就來仔細講講這個組件。

Java代碼執(zhí)行流程圖

大家通過這個流程圖聪轿,了解一下我們寫好的Java代碼是如何執(zhí)行的爷肝,其中要經(jīng)歷 類加載器 這個流程,我們就來仔細講講這里面的知識點陆错。

JVM類加載器灯抛,類加載機制詳解,看這一篇就夠了

類加載子系統(tǒng)

JVM類加載器音瓷,類加載機制詳解,看這一篇就夠了

類加載系統(tǒng)架構(gòu)圖

暫時看不懂這兩張圖沒關系绳慎,跟著 老哥 往下看

JVM類加載器纵竖,類加載機制詳解,看這一篇就夠了

類的生命周期

類的生命周期包括:加載靡砌、鏈接、初始化、使用和卸載骗炉,其中 加載 、 鏈接 蛇受、 初始化 兢仰,屬于 類加載的過程 乍丈,我們下面仔細講解。使用是指我們new對象進行使用把将,卸載指對象被垃圾回收掉了轻专。

類加載的過程

JVM類加載器,類加載機制詳解察蹲,看這一篇就夠了
  • 第一步:Loading加載

通過類的全限定名(包名 + 類名)请垛,獲取到該類的 .class

文件的二進制字節(jié)流

將二進制字節(jié)流所代表的靜態(tài)存儲結(jié)構(gòu),轉(zhuǎn)化為方法區(qū)運行時的數(shù)據(jù)結(jié)構(gòu)

在 內(nèi)存 中生成一個代表該類的 java.lang.Class 對象洽议,作為方法區(qū)這個類的各種數(shù)據(jù)的訪問入口

總結(jié): 加載二進制數(shù)據(jù)到內(nèi)存 —> 映射成jvm能識別的結(jié)構(gòu) —> 在內(nèi)存中生成class文件 宗收。

  • 第二步:Linking鏈接

鏈接是指將上面創(chuàng)建好的class類合并至Java虛擬機中,使之能夠執(zhí)行的過程亚兄,可分為 驗證 混稽、 準備 、 解析 三個階段审胚。

① 驗證(Verify)

確保class文件中的字節(jié)流包含的信息匈勋,符合當前虛擬機的要求,保證這個被加載的class類的正確性菲盾,不會危害到虛擬機的安全颓影。

② 準備(Prepare)

為類中的 靜態(tài)字段 分配內(nèi)存,并設置默認的初始值懒鉴,比如int類型初始值是0诡挂。被final修飾的static字段不會設置碎浇,因為final在編譯的時候就分配了

③ 解析(Resolve)

解析階段的目的,是將常量池內(nèi)的符號引用轉(zhuǎn)換為直接引用的過程(將常量池內(nèi)的符號引用解析成為實際引用)璃俗。如果符號引用指向一個未被加載的類奴璃,或者未被加載類的字段或方法,那么解析將觸發(fā)這個類的加載(但未必觸發(fā)這個類的鏈接以及初始化城豁。)

事實上苟穆,解析器操作往往會伴隨著 JVM 在執(zhí)行完初始化之后再執(zhí)行。 符號引用就是一組符號來描述所引用的目標唱星。符號引用的字面量形式明確定義在《Java 虛擬機規(guī)范》的Class文件格式中雳旅。直接引用就是直接指向目標的指針、相對偏移量或一個間接定位到目標的句柄间聊。

解析動作主要針對類攒盈、接口、字段哎榴、類方法型豁、接口方法、方法類型等尚蝌。對應常量池中的 CONSTANT_Class_info迎变、CONSTANT_Fieldref_info、CONSTANT_Methodref_info等飘言。

  • 第三步:initialization初始化

初始化就是執(zhí)行類的構(gòu)造器方法 init

()的過程衣形。

這個方法不需要定義,是javac編譯器自動收集類中所有類變量的賦值動作和靜態(tài)代碼塊中的語句合并來的姿鸿。

若該類具有父類泵喘, jvm 會保證父類的 init 先執(zhí)行,然后在執(zhí)行子類的 init 般妙。

類加載器的分類

JVM類加載器纪铺,類加載機制詳解,看這一篇就夠了
  • 第一個:啟動類/引導類:Bootstrap ClassLoader

這個類加載器使用C/C++語言實現(xiàn)的碟渺,嵌套在JVM內(nèi)部鲜锚,java程序無法直接操作這個類。

它用來加載Java核心類庫苫拍,如: JAVA_HOME/jre/lib/rt.jar 芜繁、 resources.jar 、 sun.boot.class.path 路徑下的包绒极,用于提供jvm運行所需的包骏令。

并不是繼承自java.lang.ClassLoader,它沒有父類加載器

它加載 擴展類加載器 和 應用程序類加載器 垄提,并成為他們的父類加載器

出于安全考慮榔袋,啟動類只加載包名為:java周拐、javax、sun開頭的類

  • 第二個:擴展類加載器:Extension ClassLoader

Java語言編寫凰兑,由 sun.misc.Launcher$ExtClassLoader 實現(xiàn)妥粟,我們可以用Java程序操作這個加載器

派生繼承自java.lang.ClassLoader,父類加載器為 啟動類加載器

從系統(tǒng)屬性: java.ext.dirs 目錄中加載類庫吏够,或者從JDK安裝目錄: jre/lib/ext 目錄下加載類庫勾给。我們就可以將我們自己的包放在以上目錄下,就會自動加載進來了锅知。

  • 第三個:應用程序類加載器:Application Classloader

Java語言編寫播急,由 sun.misc.Launcher$AppClassLoader

實現(xiàn)。

派生繼承自java.lang.ClassLoader售睹,父類加載器為 啟動類加載器

它負責加載 環(huán)境變量classpath 或者 系統(tǒng)屬性java.class.path 指定路徑下的類庫

它是程序中默認的類加載器旅择,我們Java程序中的類,都是由它加載完成的侣姆。

我們可以通過 ClassLoader#getSystemClassLoader() 獲取并操作這個加載器

  • 第四個:自定義加載器

一般情況下,以上3種加載器能滿足我們?nèi)粘5拈_發(fā)工作沉噩,不滿足時捺宗,我們還可以 自定義加載器

比如用網(wǎng)絡加載Java類,為了保證傳輸中的安全性川蒙,采用了加密操作蚜厉,那么以上3種加載器就無法加載這個類,這時候就需要 自定義加載器

自定義加載器實現(xiàn)步驟

繼承 java.lang.ClassLoader 類畜眨,重寫findClass()方法

如果沒有太復雜的需求昼牛,可以直接繼承 URLClassLoader 類,重寫 loadClass 方法康聂,具體可參考 AppClassLoader 和 ExtClassLoader 贰健。

獲取ClassLoader幾種方式

它是一個抽象類,其后所有的類加載器繼承自 ClassLoader(不包括啟動類加載器)

// 方式一:獲取當前類的 ClassLoader
clazz.getClassLoader()
// 方式二:獲取當前線程上下文的 ClassLoader
Thread.currentThread().getContextClassLoader()
// 方式三:獲取系統(tǒng)的 ClassLoader
ClassLoader.getSystemClassLoader()
// 方式四:獲取調(diào)用者的 ClassLoader
DriverManager.getCallerClassLoader()

類加載機制—雙親委派機制

jvm對class文件采用的是按需加載的方式恬汁,當需要使用該類時伶椿,jvm才會將它的class文件加載到內(nèi)存中產(chǎn)生class對象。

在加載類的時候氓侧,是采用的 雙親委派機制 脊另,即把請求交給父類處理的一種任務委派模式。

JVM類加載器约巷,類加載機制詳解偎痛,看這一篇就夠了
  • 工作原理

(1)如果一個 類加載器 接收到了 類加載 的請求,它自己不會先去加載独郎,會把這個請求委托給 父類加載器 去執(zhí)行踩麦。

(2)如果父類還存在父類加載器枚赡,則繼續(xù)向上委托,一直委托到 啟動類加載器:Bootstrap ClassLoader

(3)如果父類加載器可以完成加載任務靖榕,就返回成功結(jié)果标锄,如果父類加載失敗,就由子類自己去嘗試加載茁计,如果子類加載失敗就會拋出 ClassNotFoundException 異常料皇,這就是 雙親委派模式

  • 第三方包加載方式:反向委派機制

在Java應用中存在著很多服務提供者接口(Service Provider Interface,SPI)星压,這些接口允許第三方為它們提供實現(xiàn)践剂,如常見的 SPI 有 JDBC、JNDI等娜膘,這些 SPI 的接口屬于 Java 核心庫逊脯,一般存在rt.jar包中,由Bootstrap類加載器加載竣贪。而Bootstrap類加載器無法直接加載SPI的實現(xiàn)類军洼,同時由于雙親委派模式的存在,Bootstrap類加載器也無法反向委托AppClassLoader加載器SPI的實現(xiàn)類演怎。在這種情況下匕争,我們就需要一種特殊的類加載器來加載第三方的類庫,而線程上下文類加載器(雙親委派模型的破壞者)就是很好的選擇爷耀。

從圖可知rt.jar核心包是有Bootstrap類加載器加載的甘桑,其內(nèi)包含SPI核心接口類,由于SPI中的類經(jīng)常需要調(diào)用外部實現(xiàn)類的方法歹叮,而jdbc.jar包含外部實現(xiàn)類(jdbc.jar存在于classpath路徑)無法通過Bootstrap類加載器加載跑杭,因此只能委派線程上下文類加載器把jdbc.jar中的實現(xiàn)類加載到內(nèi)存以便SPI相關類使用。顯然這種線程上下文類加載器的加載方式破壞了“雙親委派模型”咆耿,它在執(zhí)行過程中拋棄雙親委派加載鏈模式德谅,使程序可以逆向使用類加載器,當然這也使得Java類加載器變得更加靈活萨螺。

JVM類加載器女阀,類加載機制詳解,看這一篇就夠了
  • 沙箱安全機制

自定義 String 類屑迂,但是在加載自定義 String 類的時候會率先使用引導類加載器加載浸策,而引導類加載器在加載的過程中會先加載 JDK 自帶的文件(rt.jar 包中的 javalangString.class),報錯信息說沒有 main 方法就是因為加載的 rt.jar 包中的 String 類惹盼。這樣可以保證對 Java 核心源代碼的保護庸汗,這就是沙箱安全機制。

來源:https://www.tuicool.com/articles/MVZ7zev

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末手报,一起剝皮案震驚了整個濱河市蚯舱,隨后出現(xiàn)的幾起案子改化,更是在濱河造成了極大的恐慌,老刑警劉巖枉昏,帶你破解...
    沈念sama閱讀 206,723評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件陈肛,死亡現(xiàn)場離奇詭異,居然都是意外死亡兄裂,警方通過查閱死者的電腦和手機句旱,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,485評論 2 382
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來晰奖,“玉大人谈撒,你說我怎么就攤上這事∝夷希” “怎么了啃匿?”我有些...
    開封第一講書人閱讀 152,998評論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長蛆楞。 經(jīng)常有香客問我溯乒,道長,這世上最難降的妖魔是什么豹爹? 我笑而不...
    開封第一講書人閱讀 55,323評論 1 279
  • 正文 為了忘掉前任裆悄,我火速辦了婚禮,結(jié)果婚禮上帅戒,老公的妹妹穿的比我還像新娘。我一直安慰自己崖技,他們只是感情好逻住,可當我...
    茶點故事閱讀 64,355評論 5 374
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著迎献,像睡著了一般瞎访。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上吁恍,一...
    開封第一講書人閱讀 49,079評論 1 285
  • 那天扒秸,我揣著相機與錄音,去河邊找鬼冀瓦。 笑死伴奥,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的翼闽。 我是一名探鬼主播拾徙,決...
    沈念sama閱讀 38,389評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼感局!你這毒婦竟也來了尼啡?” 一聲冷哼從身側(cè)響起暂衡,我...
    開封第一講書人閱讀 37,019評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎崖瞭,沒想到半個月后狂巢,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,519評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡书聚,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,971評論 2 325
  • 正文 我和宋清朗相戀三年唧领,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片寺惫。...
    茶點故事閱讀 38,100評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡疹吃,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出西雀,到底是詐尸還是另有隱情嫁蛇,我是刑警寧澤魂那,帶...
    沈念sama閱讀 33,738評論 4 324
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響继蜡,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜馅袁,卻給世界環(huán)境...
    茶點故事閱讀 39,293評論 3 307
  • 文/蒙蒙 一乘瓤、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧冲九,春花似錦谤草、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,289評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至灭贷,卻和暖如春温学,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背甚疟。 一陣腳步聲響...
    開封第一講書人閱讀 31,517評論 1 262
  • 我被黑心中介騙來泰國打工仗岖, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人览妖。 一個月前我還...
    沈念sama閱讀 45,547評論 2 354
  • 正文 我出身青樓轧拄,卻偏偏與公主長得像,于是被迫代替她去往敵國和親讽膏。 傳聞我的和親對象是個殘疾皇子紧帕,可洞房花燭夜當晚...
    茶點故事閱讀 42,834評論 2 345