摘要:如果需要支持類的動(dòng)態(tài)加載或需要對(duì)編譯后的字節(jié)碼文件進(jìn)行解密操作等,就需要與類加載器打交道了。雙親委派模型,雙親委派模型,約定類加載器的加載機(jī)制。任何之類的字節(jié)碼都無法調(diào)用方法,因?yàn)樵摲椒ㄖ荒茉陬惣虞d的過程中由調(diào)用。
jvm系列
垃圾回收基礎(chǔ)
JVM的編譯策略
GC的三大基礎(chǔ)算法
GC的三大高級(jí)算法
GC策略的評(píng)價(jià)指標(biāo)
JVM信息查看
GC通用日志解讀
jvm的card table數(shù)據(jù)結(jié)構(gòu)
Java類初始化順序
Java對(duì)象結(jié)構(gòu)及大小計(jì)算
Java的類加載機(jī)制
Java對(duì)象分配簡要流程
年老代過大有什么影響
Survivor空間溢出實(shí)例
關(guān)于Object=null
Java線程與Xss
序本文主要講述Java類的加載機(jī)制,主要包括類加載器、加載過程、初始化時(shí)機(jī)。
一、類加載器 1、ClassLoader抽象類類加載器的任務(wù)就是根據(jù)一個(gè)類的全限定名來讀取此類的二進(jìn)制字節(jié)流到JVM內(nèi)部,然后轉(zhuǎn)換為一個(gè)與目標(biāo)類對(duì)應(yīng)的java.lang.Class對(duì)象實(shí)例。
如果需要支持類的動(dòng)態(tài)加載或需要對(duì)編譯后的字節(jié)碼文件進(jìn)行解密操作等,就需要與類加載器打交道了。
BootstrapClassLoader,由C++編寫嵌套在JVM內(nèi)部,負(fù)責(zé)加載“JAVA_HOME/lib”目錄中的所有類型,或者由“-Xbootclasspath”指定路徑中的所有類型。
ExtClassLoader和AppClassLoader都繼承至ClassLoader抽象類,由Java編寫。
ExtClassLoader負(fù)責(zé)加載“JAVA_HOME/lib/ext”目錄下的所有類型。
AppClassLoader負(fù)責(zé)加載ClassPath目錄中的所有類型。
defineClass方法將字節(jié)碼的byte數(shù)組轉(zhuǎn)換為一個(gè)類的Class對(duì)象實(shí)例,如果希望在類被加載到JVM內(nèi)部時(shí)就被鏈接,那么可以調(diào)用resolveClass方法。
2、雙親委派模型Parents Delegation Model,雙親委派模型,約定類加載器的加載機(jī)制。
當(dāng)一個(gè)類加載器接收到一個(gè)類加載的任務(wù)時(shí),不會(huì)立即展開加載,而是將加載任務(wù)委派給它的超類加載器去執(zhí)行,每一層的類都采用相同的方式,直至委派給最頂層的啟動(dòng)類加載器為止。如果超類加載器無法加載委派給它的類,便將類的加載任務(wù)退回給下一級(jí)類加載器去執(zhí)行加載。
雙親委派模型的工作過程是:如果一個(gè)類加載器收到了類加載的請(qǐng)求,它首先不會(huì)自己去嘗試加載這個(gè)類,而是把這個(gè)請(qǐng)求委派給父類加載器去完成,每一個(gè)層次的類加載器都是如此,因此所有的加載請(qǐng)求最終都應(yīng)該傳送到頂層的啟動(dòng)類加載器中,只有當(dāng)父加載器反饋?zhàn)约簾o法完成這個(gè)加載請(qǐng)求(它的搜索范圍中沒有找到所需的類)時(shí),子加載器才會(huì)嘗試自己去加載。
使用這種方式的好處是:能夠有效確保一個(gè)類的全局唯一性,當(dāng)程序中出現(xiàn)多個(gè)全限定名相同的類時(shí),類加載器在執(zhí)行加載時(shí),始終只會(huì)加載其中的某一個(gè)類。
使用雙親委派模型來組織類加載器之間的關(guān)系,有一個(gè)顯而易見的好處就是Java類隨著它的類加載器一起具備了一種帶有優(yōu)先級(jí)的層次關(guān)系。例如類java.lang.Object,它存放在rt.jar之中,無論哪一個(gè)類加載器要加載這個(gè)類,最終都是委派給處于模型最頂端的啟動(dòng)類加載器進(jìn)行加載,因此Object類在程序的各種類加載器環(huán)境中都是同一個(gè)類。相反,如果沒有使用雙親委派模型,由各個(gè)類加載器自行去加載的話,如果用戶自己編寫了一個(gè)稱為java.lang.Object的類,并放在程序的Class-Path中,那系統(tǒng)中將會(huì)出現(xiàn)多個(gè)不同的Object類,Java類型體系中最基礎(chǔ)的行為也就無法保證,應(yīng)用程序也將會(huì)變得一片混亂。如果自己去編寫一個(gè)與rt.jar類庫中已有類重名的Java類,將會(huì)發(fā)現(xiàn)可以正常編譯,但永遠(yuǎn)無法被加載運(yùn)行。
雙親委派模型對(duì)于保證Java程序的穩(wěn)定運(yùn)作很重要,但它的實(shí)現(xiàn)卻非常簡單,實(shí)現(xiàn)雙親委派的代碼都集中在java.lang.ClassLoader的loadClass()方法之中,邏輯清晰易懂:先檢查是否已經(jīng)被加載過,若沒有加載則調(diào)用父加載器的loadClass()方法,若父加載器為空則默認(rèn)使用啟動(dòng)類加載器作為父加載器。如果父類加載失敗,拋出ClassNotFoundException異常后,再調(diào)用自己的findClass()方法進(jìn)行加載。
雙親委派機(jī)制只是Java虛擬機(jī)規(guī)范建議采用的加載機(jī)制,實(shí)際在tomcat中,類加載器所采用的加載機(jī)制與傳統(tǒng)的雙親委派模型有一定的區(qū)別,當(dāng)缺省的類加載器接收到一個(gè)類的加載任務(wù)時(shí),首先會(huì)去由它自行加載,當(dāng)它加載失敗時(shí),才會(huì)將類的加載任務(wù)委派給它的超類加載器去執(zhí)行。
3、自定義類加載器程序中如果沒有顯式指定類加載器的話,默認(rèn)是AppClassLoader來加載,它負(fù)責(zé)加載ClassPath目錄中的所有類型,如果被加載的類型并沒有在ClassPath目錄中時(shí),拋出java.lang.ClassNotFoundException異常。
一般是繼承ClassLoader,如果要符合雙親委派規(guī)范,則重寫findClass方法;要破壞的話,重寫loadClass方法。
雙親委派模型的第一次“被破壞”其實(shí)發(fā)生在雙親委派模型出現(xiàn)之前——即JDK 1.2發(fā)布之前。由于雙親委派模型在JDK 1.2之后才被引入,而類加載器和抽象類java.lang.ClassLoader則在JDK1.0時(shí)代就已經(jīng)存在,面對(duì)已經(jīng)存在的用戶自定義類加載器的實(shí)現(xiàn)代碼,Java設(shè)計(jì)者引入雙親委派模型時(shí)不得不做出一些妥協(xié)。
為了向前兼容,JDK 1.2之后的java.lang.ClassLoader添加了一個(gè)新的protected方法findClass(),在此之前,用戶去繼承java.lang.ClassLoader的唯一目的就是為了重寫loadClass()方法,因?yàn)樘摂M機(jī)在進(jìn)行類加載的時(shí)候會(huì)調(diào)用加載器的私有方法loadClassInternal(),而這個(gè)方法的唯一邏輯就是去調(diào)用自己的load-Class()。
上一節(jié)我們已經(jīng)看過loadClass()方法的代碼,雙親委派的具體邏輯就實(shí)現(xiàn)在這個(gè)方法之中,JDK1.2之后已不提倡用戶再去覆蓋loadClass()方法,而應(yīng)當(dāng)把自己的類加載邏輯寫到findClass()方法中,在loadClass()方法的邏輯里如果父類加載失敗,則會(huì)調(diào)用自己的findClass()方法來完成加載,這樣就可以保證新寫出來的類加載器是符合雙親委派規(guī)則的。
二、類加載過程一個(gè)完整的類加載過程必須經(jīng)歷加載、連接、初始化這三個(gè)步驟:
1、加載簡單的說,類加載階段就是由類加載器負(fù)責(zé)根據(jù)一個(gè)類的全限定名來讀取此類的二進(jìn)制字節(jié)流到JVM內(nèi)部,并存儲(chǔ)在運(yùn)行時(shí)內(nèi)存區(qū)的方法區(qū),然后將其轉(zhuǎn)換為一個(gè)與目標(biāo)類型對(duì)應(yīng)的java.lang.Class對(duì)象實(shí)例(Java虛擬機(jī)規(guī)范并沒有明確要求一定要存儲(chǔ)在堆區(qū)中,只是hotspot選擇將Class對(duì)象存儲(chǔ)在方法區(qū)中),這個(gè)Class對(duì)象在日后就會(huì)作為方法區(qū)中該類的各種數(shù)據(jù)的訪問入口。
2、連接連接階段要做的是將加載到JVM中的二進(jìn)制字節(jié)流的類數(shù)據(jù)信息合并到JVM的運(yùn)行時(shí)狀態(tài)中,經(jīng)由驗(yàn)證、準(zhǔn)備、解析三個(gè)階段。
(1)驗(yàn)證階段
驗(yàn)證類數(shù)據(jù)信息是否符合JVM規(guī)范,是否是一個(gè)有效的字節(jié)碼文件,驗(yàn)證內(nèi)容涵蓋了類數(shù)據(jù)信息的格式驗(yàn)證、語義分析、操作驗(yàn)證等
格式驗(yàn)證:驗(yàn)證是否符合class文件規(guī)范,比如以0xCAFEBABE開頭,大小版本號(hào)等
語義驗(yàn)證:
a、檢查一個(gè)被標(biāo)記為final的類型是否包含派生類
b、檢查一個(gè)類中的final方法是否被派生類進(jìn)行重寫
c、確保超類與派生類之間沒有不兼容的一些方法聲明(比如方法簽名相同,但方法的返回值不同)
操作驗(yàn)證:
在操作數(shù)棧中的數(shù)據(jù)必須進(jìn)行正確的操作,對(duì)常量池中的各種符號(hào)引用執(zhí)行驗(yàn)證(通常在解析階段執(zhí)行,檢查是否能通過符號(hào)引用中描述的全限定名定位到指定類型上,以及類成員信息的訪問修飾符是否允許訪問等)。
(2)準(zhǔn)備階段
為類中的所有靜態(tài)變量分配內(nèi)存空間,并為其設(shè)置一個(gè)初始值(由于還沒有產(chǎn)生對(duì)象,實(shí)例變量將不再此操作范圍內(nèi))
(3)解析階段
將常量池中所有的符號(hào)引用轉(zhuǎn)為直接引用(得到類或者字段、方法在內(nèi)存中的指針或者偏移量,以便直接調(diào)用該方法)。這個(gè)階段可以在初始化之后再執(zhí)行。
將一個(gè)類中所有被static關(guān)鍵字標(biāo)識(shí)的代碼統(tǒng)一執(zhí)行一遍,如果執(zhí)行的是靜態(tài)變量,那么就會(huì)使用用戶指定的值覆蓋之前在準(zhǔn)備階段設(shè)置的初始值;如果執(zhí)行的是static代碼塊,那么在初始化階段,JVM就會(huì)執(zhí)行static代碼塊中定義的所有操作。
所有類變量初始化語句和靜態(tài)代碼塊都會(huì)在編譯時(shí)被前端編譯器放在收集器里頭,存放到一個(gè)特殊的方法中,這個(gè)方法就是
如果超類還沒有被初始化,那么優(yōu)先對(duì)超類初始化,但在
JVM必須確保一個(gè)類在初始化的過程中,如果是多線程需要同時(shí)初始化它,僅僅只能允許其中一個(gè)線程對(duì)其執(zhí)行初始化操作,其余線程必須等待,只有在活動(dòng)線程執(zhí)行完對(duì)類的初始化操作之后,才會(huì)通知正在等待的其他線程。
只有那些需要執(zhí)行java代碼來為類變量執(zhí)行賦值操作的類型在編譯之后才會(huì)在字節(jié)碼中存在生成的
(1)為一個(gè)類型創(chuàng)建一個(gè)新的對(duì)象實(shí)例時(shí)(比如new、反射、序列化)
(2)調(diào)用一個(gè)類型的靜態(tài)方法時(shí)(即在字節(jié)碼中執(zhí)行invokestatic指令)
(3)調(diào)用一個(gè)類型或接口的靜態(tài)字段,或者對(duì)這些靜態(tài)字段執(zhí)行賦值操作時(shí)(即在字節(jié)碼中,執(zhí)行g(shù)etstatic或者putstatic指令),不過用final修飾的靜態(tài)字段除外,它被初始化為一個(gè)編譯時(shí)常量表達(dá)式
(4)調(diào)用JavaAPI中的反射方法時(shí)(比如調(diào)用java.lang.Class中的方法,或者java.lang.reflect包中其他類的方法)
(5)初始化一個(gè)類的派生類時(shí)(Java虛擬機(jī)規(guī)范明確要求初始化一個(gè)類時(shí),它的超類必須提前完成初始化操作,接口例外)
(6)JVM啟動(dòng)包含main方法的啟動(dòng)類時(shí)。
數(shù)組本身并不是由類加載器負(fù)責(zé)創(chuàng)建,而是由JVM在運(yùn)行時(shí)根據(jù)需要而直接創(chuàng)建的,但數(shù)組的元素類型仍然需要依靠類加載器去創(chuàng)建。
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://m.specialneedsforspecialkids.com/yun/65610.html
摘要:當(dāng)前類加載器和所有父類加載器都無法加載該類時(shí),拋出異常。加載兩份相同的對(duì)象的情況和不屬于父子類加載器關(guān)系,并且各自都加載了同一個(gè)類。類加載機(jī)制與接口當(dāng)虛擬機(jī)初始化一個(gè)類時(shí),不會(huì)初始化該類實(shí)現(xiàn)的接口。 類加載機(jī)制 概念 類加載器把class文件中的二進(jìn)制數(shù)據(jù)讀入到內(nèi)存中,存放在方法區(qū),然后在堆區(qū)創(chuàng)建一個(gè)java.lang.Class對(duì)象,用來封裝類在方法區(qū)內(nèi)的數(shù)據(jù)結(jié)構(gòu)。 1、加載: 查...
摘要:作用負(fù)責(zé)將加載到中審查每個(gè)類由誰加載父優(yōu)先的等級(jí)加載機(jī)制將字節(jié)碼重新解析成統(tǒng)一要求的對(duì)象格式類結(jié)構(gòu)分析為了更好的理解類的加載機(jī)制,我們來深入研究一下和他的方法。就算兩個(gè)是同一份字節(jié)碼,如果被兩個(gè)不同的實(shí)例所加載,也會(huì)認(rèn)為它們是兩個(gè)不同。 申明:本文首發(fā)于 詳細(xì)深入分析 ClassLoader 工作機(jī)制 ,如有轉(zhuǎn)載,注明原出處即可,謝謝配合。 什么是 ClassLoader ? 大家...
摘要:前面提到,對(duì)于數(shù)組類來說,它并沒有對(duì)應(yīng)的字節(jié)流,而是由虛擬機(jī)直接生成的。對(duì)于其他的類來說,虛擬機(jī)則需要借助類加載器來完成查找字節(jié)流的過程。驗(yàn)證階段的目的,在于確保被加載類能夠滿足虛擬機(jī)的約束條件。 Java 虛擬機(jī)將字節(jié)流轉(zhuǎn)化為 Java 類的過程。這個(gè)過程可分為加載、鏈接以及初始化 三大步驟。 加載是指查找字節(jié)流,并且據(jù)此創(chuàng)建類的過程。加載需要借助類加載器,在 Java 虛擬機(jī)中,類...
摘要:類加載器三杰有三類,分別是以及。線程上下文類加載器線程上下文類加載器可以不遵循雙親委派機(jī)制。免費(fèi)領(lǐng)取驗(yàn)證碼內(nèi)容安全短信發(fā)送直播點(diǎn)播體驗(yàn)包及云服務(wù)器等套餐更多網(wǎng)易技術(shù)產(chǎn)品運(yùn)營經(jīng)驗(yàn)分享請(qǐng)?jiān)L問網(wǎng)易云社區(qū)。文章來源網(wǎng)易云社區(qū) 本文由作者張遠(yuǎn)道授權(quán)網(wǎng)易云社區(qū)發(fā)布。 類加載器三杰 jvm有三類classloader,分別是bootstrap classloader,extended classlo...
摘要:當(dāng)程序使用某個(gè)類時(shí),如果該類還沒被初始化,加載到內(nèi)存中,則系統(tǒng)會(huì)通過加載連接初始化三個(gè)過程來對(duì)該類進(jìn)行初始化。一旦一個(gè)類被加載到中之后,就不會(huì)再次載入了。它既可以從本地文件系統(tǒng)獲取二進(jìn)制文件來加載類,也可以遠(yuǎn)程主機(jī)獲取二進(jìn)制文件來加載類。 當(dāng)程序使用某個(gè)類時(shí),如果該類還沒被初始化,加載到內(nèi)存中,則系統(tǒng)會(huì)通過加載、連接、初始化三個(gè)過程來對(duì)該類進(jìn)行初始化。該過程就被稱為類的初始化 類加載 ...
閱讀 3903·2021-11-22 13:54
閱讀 2674·2021-09-30 09:48
閱讀 2360·2021-09-28 09:36
閱讀 3113·2021-09-22 15:26
閱讀 1342·2019-08-30 15:55
閱讀 2509·2019-08-30 15:54
閱讀 1425·2019-08-30 14:17
閱讀 2340·2019-08-28 18:25