這篇文章主要介紹了JVM核心之JVM運(yùn)行和類加載,具有一定借鑒價(jià)值,感興趣的朋友可以參考下,希望大家閱讀完這篇文章之后大有收獲,下面讓小編帶著大家一起了解一下。
10余年的定南網(wǎng)站建設(shè)經(jīng)驗(yàn),針對設(shè)計(jì)、前端、開發(fā)、售后、文案、推廣等六對一服務(wù),響應(yīng)快,48小時(shí)及時(shí)工作處理。全網(wǎng)營銷推廣的優(yōu)勢是能夠根據(jù)用戶設(shè)備顯示端的尺寸不同,自動(dòng)調(diào)整定南建站的顯示方式,使網(wǎng)站能夠適用不同顯示終端,在瀏覽器中調(diào)整網(wǎng)站的寬度,無論在任何一種瀏覽器上瀏覽網(wǎng)站,都能展現(xiàn)優(yōu)雅布局與設(shè)計(jì),從而大程度地提升瀏覽體驗(yàn)。創(chuàng)新互聯(lián)從事“定南網(wǎng)站設(shè)計(jì)”,“定南網(wǎng)站推廣”以來,每個(gè)客戶項(xiàng)目都認(rèn)真落實(shí)執(zhí)行。
關(guān)于JVM運(yùn)行時(shí)數(shù)據(jù)區(qū)
JVM運(yùn)行時(shí)數(shù)據(jù)區(qū)
關(guān)于類加載
class文件加載至內(nèi)存,鏈接(校驗(yàn)、解析),初始化;最終形成JVM可以直接使用的JAVA類型的過程。
加載:在方法區(qū)形成類的運(yùn)行時(shí)數(shù)據(jù)結(jié)構(gòu);在堆里面形成該類的Class對象,作為訪問方法區(qū)的入口。
加載
鏈接:class文件是否存在問題;一些符號引號替換成直接引用。
初始化:初始化一個(gè)類,先初始化它的父類。虛擬機(jī)會(huì)保證一個(gè)類的初始化在多線程環(huán)境中被正確加鎖和同步。
要使用類A,必須先加載類A;加載類A,就會(huì)把靜態(tài)變量、靜態(tài)塊合并初始化,然后在調(diào)用構(gòu)造器。注意類的加載和初始化,只有一次。
關(guān)于類加載器
上文已經(jīng)說了,類加載器的作用就是:將class文件的字節(jié)碼內(nèi)容加載到內(nèi)存中,并將這些靜態(tài)數(shù)據(jù)轉(zhuǎn)化成方法區(qū)中的運(yùn)行時(shí)數(shù)據(jù)結(jié)構(gòu),在堆中生成一個(gè)代表這個(gè)類的Class對象,作為方法區(qū)類數(shù)據(jù)的訪問入口。
類加載器的層次結(jié)構(gòu)
引導(dǎo)類加載器bootstrap classloader
加載JAVA核心庫($JAVA_HOME/jre/lib/rt.jar),原生代碼實(shí)現(xiàn)(C++),并不繼承自java.lang.ClassLoader。
擴(kuò)展類加載器extensions classloader
JAVA可以提供一個(gè)擴(kuò)展目錄($JAVA_HOME/jre/ext/*.jar)來加載Java類。
由sun.misc.Launcher.ExtClassLoader實(shí)現(xiàn)
應(yīng)用程序類加載器application classloader(也稱系統(tǒng)類加載器)
一般來說,JAVA應(yīng)用的類由它加載,即加載路徑是classpath下的路徑。
由sun.misc.Launcher.AppClassLoader實(shí)現(xiàn)。
自定義類加載器
開發(fā)人員繼承java.lang.ClassLoader實(shí)現(xiàn)自己的類加載器
類加載器的層次結(jié)構(gòu)
關(guān)于java.lang.ClassLoader
ClassLoader的基本職責(zé)就是:
第一,根據(jù)指定的類名稱,找到或者生成對應(yīng)的字節(jié)碼,并根據(jù)字節(jié)碼生成class對象
第二,加載JAVA應(yīng)用所需的資源,如配置文件等。
ClassLoader的組合模式
組合模式為雙親委派機(jī)制提供支持
demo:
類加載器的層次
引導(dǎo)類加載器是原生代碼實(shí)現(xiàn),我們獲取不到,所以是null。
ClassLoader重要API
getParent():該類加載器的父類加載器
loadClass(String name):加載名稱為name的類,并返回Class實(shí)例。
加載順序是:先交給擴(kuò)展類加載器加載,如果加載不到,交給引導(dǎo)類加載器加載,加載不到,交給自己去加載,如果自己也加載不到,那么ClassNotFoundException。【雙親委派機(jī)制】 如果要改變類的加載順序,那么可以override該方法。
findClass(String name),不是加載,僅僅是查找而已
findLoadedClass(String name),查找已經(jīng)被加載過的
defineClass(String name,byte[] b, int off ,int len),可以把字節(jié)數(shù)組的內(nèi)容轉(zhuǎn)換成JAVA類,并會(huì)返回Class實(shí)例。
類加載器的代理模式:雙親委派機(jī)制
類加載器的代理模式:就是把加載指定類的過程交給其他加載器。
JAVA默認(rèn)使用的類加載器代理模式是:雙親委派機(jī)制。
雙親委派機(jī)制:
就是某個(gè)特定的類加載器接到加載類的請求時(shí),首先將加載任務(wù)委托給父類加載器,依次追溯,比如說從應(yīng)用加載器委托給擴(kuò)展類加載器,從擴(kuò)展類加載器委托給引導(dǎo)類加載器。這種委托,直至委托到層次最高的類加載器,即引導(dǎo)類加載器,如果委托的父類加載器可以完成加載任務(wù),那么成功返回;只有父類加載器無法完成時(shí),才去自己加載。
可以看出雙親委派機(jī)制的意思就是優(yōu)先父類加載器加載!
試想如果我們定義了一個(gè)java.lang.String類,根據(jù)雙親委派機(jī)制,那么JDK只會(huì)加載它自己的String。這顯然保證了Java核心庫的類型安全。
雙親委派機(jī)制不是唯一的選擇
雖然JDK默認(rèn)的類加載機(jī)制是雙親委派機(jī)制,但是并不是所有都采用,比如有些服務(wù)器,如Tomcat,雖然也采用代理的方式加載,但是加載順序卻恰恰和雙親委派機(jī)制相反,它是首先嘗試加載這個(gè)類,只有加載不到的情況下,才去讓父類加載器代理加載。
為什么會(huì)這樣呢,不是說雙親委派很安全么?
其實(shí)就是在安全,和靈活方面進(jìn)行取舍!
寫一個(gè)自定義類加載器
MyClassLoader:
自定義類加載器
重寫findClass:
findClass
Test:
感謝你能夠認(rèn)真閱讀完這篇文章,希望小編分享的“JVM核心之JVM運(yùn)行和類加載”這篇文章對大家有幫助,同時(shí)也希望大家多多支持創(chuàng)新互聯(lián),關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,更多相關(guān)知識等著你來學(xué)習(xí)!