日本综合一区二区|亚洲中文天堂综合|日韩欧美自拍一区|男女精品天堂一区|欧美自拍第6页亚洲成人精品一区|亚洲黄色天堂一区二区成人|超碰91偷拍第一页|日韩av夜夜嗨中文字幕|久久蜜综合视频官网|精美人妻一区二区三区

RELATEED CONSULTING
相關(guān)咨詢(xún)
選擇下列產(chǎn)品馬上在線(xiàn)溝通
服務(wù)時(shí)間:8:30-17:00
你可能遇到了下面的問(wèn)題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營(yíng)銷(xiāo)解決方案
Java類(lèi)加載機(jī)制及類(lèi)加載器詳解

本文轉(zhuǎn)載自微信公眾號(hào)「愛(ài)寫(xiě)B(tài)ug的麥洛」,作者麥洛。轉(zhuǎn)載本文請(qǐng)聯(lián)系愛(ài)寫(xiě)B(tài)ug的麥洛公眾號(hào)。

網(wǎng)站建設(shè)哪家好,找創(chuàng)新互聯(lián)建站!專(zhuān)注于網(wǎng)頁(yè)設(shè)計(jì)、網(wǎng)站建設(shè)、微信開(kāi)發(fā)、重慶小程序開(kāi)發(fā)、集團(tuán)企業(yè)網(wǎng)站建設(shè)等服務(wù)項(xiàng)目。為回饋新老客戶(hù)創(chuàng)新互聯(lián)還提供了通道免費(fèi)建站歡迎大家使用!

一、類(lèi)加載機(jī)制

1.什么是類(lèi)加載?

熟悉java開(kāi)發(fā)的同學(xué)都知道,我們?nèi)粘K鶎?xiě)的代碼都被保存到.java文件中。這些".java"文件經(jīng)過(guò)Java編譯器編譯成拓展名為".class"的文件,".class"文件中保存著Java代碼經(jīng)轉(zhuǎn)換后的虛擬機(jī)指令,當(dāng)需要使用某個(gè)類(lèi)時(shí),虛擬機(jī)將會(huì)加載它的".class"文件,并創(chuàng)建對(duì)應(yīng)的class對(duì)象,將class文件加載到虛擬機(jī)的內(nèi)存,這個(gè)過(guò)程稱(chēng)為類(lèi)加載

2.類(lèi)加載的過(guò)程

加載,驗(yàn)證,準(zhǔn)備,解析,初始化,使用和卸載。其中驗(yàn)證,準(zhǔn)備,解析3個(gè)部分統(tǒng)稱(chēng)為連接。

這7個(gè)階段發(fā)生順序如下圖:

其中加載,驗(yàn)證,準(zhǔn)備,解析及初始化是屬于類(lèi)加載機(jī)制中的步驟。注意此處的加載不等同于類(lèi)加載,大家兩張圖對(duì)比看著理解。

3.觸發(fā)類(lèi)加載的條件

①.遇到new,getstatic,putstatic或invokestatic這4條字節(jié)碼指令時(shí),如果類(lèi)沒(méi)有進(jìn)行過(guò)初始化,則需要先觸發(fā)初始化。生成這4條指令的最常見(jiàn)的Java代碼場(chǎng)景是:使用new關(guān)鍵字實(shí)例化對(duì)象的時(shí)候,讀取或設(shè)置一個(gè)類(lèi)的靜態(tài)字段的時(shí)候(被final修飾,已在編譯期把結(jié)果放入常量池的靜態(tài)字段除外),以及調(diào)用一個(gè)類(lèi)的靜態(tài)方法的時(shí)候。

②.使用java.lang.reflect包的方法對(duì)類(lèi)進(jìn)行反射調(diào)用的時(shí)候。

③.當(dāng)初始化一個(gè)類(lèi)的時(shí)候,發(fā)現(xiàn)其父類(lèi)還沒(méi)有進(jìn)行過(guò)初始化,則需要先出發(fā)父類(lèi)的初始化。

④.當(dāng)虛擬機(jī)啟動(dòng)時(shí),用戶(hù)需要指定一個(gè)要執(zhí)行的主類(lèi)(包含main()方法的那個(gè)類(lèi)),虛擬機(jī)會(huì)先初始化這個(gè)主類(lèi)。

⑤.當(dāng)使用JDK1.7的動(dòng)態(tài)語(yǔ)言支持時(shí),如果一個(gè)java.lang.invoke.MethodHandle實(shí)例最后的解析結(jié)果REF_getStatic,REF_putStatic,REF_invokeStatic的方法句柄,并且這個(gè)方法句柄所對(duì)應(yīng)的類(lèi)沒(méi)有進(jìn)行初始化,則需要先出發(fā)初始化。

4.類(lèi)加載的具體過(guò)程

加載:

①.通過(guò)一個(gè)類(lèi)的全限定名來(lái)獲取定義此類(lèi)的二進(jìn)制字節(jié)流

②.將這個(gè)字節(jié)流所代表的靜態(tài)存儲(chǔ)結(jié)構(gòu)轉(zhuǎn)換為方法區(qū)內(nèi)的運(yùn)行時(shí)數(shù)據(jù)結(jié)構(gòu)

③.在內(nèi)存中生成一個(gè)代表這個(gè)類(lèi)的java.lang.Class對(duì)象,作為方法區(qū)這個(gè)類(lèi)的各種數(shù)據(jù)的訪(fǎng)問(wèn)入口。驗(yàn)證:

是連接階段的第一步,目的是為了確保Class文件的字節(jié)流中包含的信息符合當(dāng)前虛擬機(jī)的要求,并且不會(huì)危害虛擬機(jī)自身的安全。

包含四個(gè)階段的校驗(yàn)動(dòng)作

a.文件格式驗(yàn)證 驗(yàn)證字節(jié)流是否符合Class文件格式的規(guī)范,并且能被當(dāng)前版本的虛擬機(jī)處理。b.元數(shù)據(jù)驗(yàn)證 對(duì)類(lèi)的元數(shù)據(jù)信息進(jìn)行語(yǔ)義校驗(yàn),是否不存在不符合Java語(yǔ)言規(guī)范的元數(shù)據(jù)信息 c.字節(jié)碼驗(yàn)證 最復(fù)雜的一個(gè)階段,主要目的是通過(guò)數(shù)據(jù)流和控制流分析,確定程序語(yǔ)義是合法的,符合邏輯的。對(duì)類(lèi)的方法體進(jìn)行校驗(yàn)分析,保證被校驗(yàn)類(lèi)的方法在運(yùn)行時(shí)不會(huì)做出危害虛擬機(jī)安全的事件。d.符號(hào)引用驗(yàn)證 最后一個(gè)階段的校驗(yàn)發(fā)生在虛擬機(jī)將符號(hào)引用轉(zhuǎn)換為直接引用的時(shí)候,這個(gè)轉(zhuǎn)換動(dòng)作將在連接的第三個(gè)階段——解析階段中發(fā)生。符號(hào)驗(yàn)證的目的是確保解析動(dòng)作能正常進(jìn)行。

準(zhǔn)備:準(zhǔn)備階段是正式為類(lèi)變量分配內(nèi)存并設(shè)置類(lèi)變量初始值的階段。這些變量所使用的內(nèi)存都將在方法區(qū)中分配。只包括類(lèi)變量。初始值“通常情況”下是數(shù)據(jù)類(lèi)型的零值。

“特殊情況”下,如果類(lèi)字段的字段屬性表中存在ConstantValue屬性,那么在準(zhǔn)備階段變量的值就會(huì)被初始化為ConstantValue屬性所指定的值。

解析:虛擬機(jī)將常量池內(nèi)的符號(hào)引用替換為直接引用的過(guò)程。

“動(dòng)態(tài)解析”的含義就是必須等到程序?qū)嶋H運(yùn)行到這條指令的時(shí)候,解析動(dòng)作才能進(jìn)行。相對(duì)的,其余可觸發(fā)解析的指令都是“靜態(tài)”的,可以在剛剛完成加載階段,還沒(méi)有開(kāi)始執(zhí)行代碼時(shí)就進(jìn)行解析。

初始化:

類(lèi)加載過(guò)程中的最后一步。

初始化階段是執(zhí)行類(lèi)構(gòu)造器()方法的過(guò)程。

()方法是由編譯器自動(dòng)收集類(lèi)中的所有類(lèi)變量的賦值動(dòng)作和靜態(tài)語(yǔ)句塊中的語(yǔ)句合并產(chǎn)生的。

()與類(lèi)的構(gòu)造函數(shù)不同,它不需要顯示地調(diào)用父類(lèi)構(gòu)造器,虛擬機(jī)會(huì)保證在子類(lèi)的()方法執(zhí)行之前,父類(lèi)的()方法已經(jīng)執(zhí)行完畢。

簡(jiǎn)單地說(shuō),初始化就是對(duì)類(lèi)變量進(jìn)行賦值及執(zhí)行靜態(tài)代碼塊。

二、類(lèi)加載器

通過(guò)上述的了解,我們已經(jīng)知道了類(lèi)加載機(jī)制的大概流程及各個(gè)部分的功能。其中加載部分的功能是將類(lèi)的class文件讀入內(nèi)存,并為之創(chuàng)建一個(gè)java.lang.Class對(duì)象。這部分功能就是由類(lèi)加載器來(lái)實(shí)現(xiàn)的。

1.類(lèi)加載器分類(lèi):

不同的類(lèi)加載器負(fù)責(zé)加載不同的類(lèi)。主要分為兩類(lèi)。

啟動(dòng)類(lèi)加載器(Bootstrap ClassLoader):由C++語(yǔ)言實(shí)現(xiàn)(針對(duì)HotSpot),負(fù)責(zé)將存放在 \lib目錄或-Xbootclasspath參數(shù)指定的路徑中的類(lèi)庫(kù)加載到內(nèi)存中,即負(fù)責(zé)加載Java的核心類(lèi)。

擴(kuò)展類(lèi)加載器(Extension ClassLoader):負(fù)責(zé)加載 \lib\ext目錄或java.ext.dirs系統(tǒng)變量指定的路徑中的所有類(lèi)庫(kù),即負(fù)責(zé)加載Java擴(kuò)展的核心類(lèi)之外的類(lèi)。

應(yīng)用程序類(lèi)加載器(Application ClassLoader):負(fù)責(zé)加載用戶(hù)類(lèi)路徑(classpath)上的指定類(lèi)庫(kù),我們可以直接使用這個(gè)類(lèi)加載器,通過(guò)ClassLoader.getSystemClassLoader()方法直接獲取。一般情況,如果我們沒(méi)有自定義類(lèi)加載器默認(rèn)就是用這個(gè)加載器。

其他類(lèi)加載器:由Java語(yǔ)言實(shí)現(xiàn),繼承自抽象類(lèi)ClassLoader。

下面我們來(lái)具體了解上述幾個(gè)類(lèi)加載器實(shí)現(xiàn)類(lèi)加載過(guò)程時(shí)相互配合協(xié)作的流程。

2.雙親委派模型

雙親委派模型的工作流程是:如果一個(gè)類(lèi)加載器收到了類(lèi)加載的請(qǐng)求,它首先不會(huì)自己去嘗試加載這個(gè)類(lèi),而是把請(qǐng)求委托給父加載器去完成,依次向上,因此,所有的類(lèi)加載請(qǐng)求最終都應(yīng)該被傳遞到頂層的啟動(dòng)類(lèi)加載器中,只有當(dāng)父加載器在它的搜索范圍中沒(méi)有找到所需的類(lèi)時(shí),即無(wú)法完成該加載,子加載器才會(huì)嘗試自己去加載該類(lèi)。

這樣的好處是不同層次的類(lèi)加載器具有不同優(yōu)先級(jí),比如所有Java對(duì)象的超級(jí)父類(lèi)java.lang.Object,位于rt.jar,無(wú)論哪個(gè)類(lèi)加載器加載該類(lèi),最終都是由啟動(dòng)類(lèi)加載器進(jìn)行加載,保證安全。即使用戶(hù)自己編寫(xiě)一個(gè)java.lang.Object類(lèi)并放入程序中,雖能正常編譯,但不會(huì)被加載運(yùn)行,保證不會(huì)出現(xiàn)混亂。

3.雙親委派模型的代碼實(shí)現(xiàn)

ClassLoader中l(wèi)oadClass方法實(shí)現(xiàn)了雙親委派模型

 
 
 
 
  1. protected Class loadClass(String name, boolean resolve) 
  2.     throws ClassNotFoundException 
  3.     synchronized (getClassLoadingLock(name)) { 
  4.         //檢查該類(lèi)是否已經(jīng)加載過(guò) 
  5.         Class c = findLoadedClass(name); 
  6.         if (c == null) { 
  7.             //如果該類(lèi)沒(méi)有加載,則進(jìn)入該分支 
  8.             long t0 = System.nanoTime(); 
  9.             try { 
  10.                 if (parent != null) { 
  11.                     //當(dāng)父類(lèi)的加載器不為空,則通過(guò)父類(lèi)的loadClass來(lái)加載該類(lèi) 
  12.                     c = parent.loadClass(name, false); 
  13.                 } else { 
  14.                     //當(dāng)父類(lèi)的加載器為空,則調(diào)用啟動(dòng)類(lèi)加載器來(lái)加載該類(lèi) 
  15.                     c = findBootstrapClassOrNull(name); 
  16.                 } 
  17.             } catch (ClassNotFoundException e) { 
  18.                 //非空父類(lèi)的類(lèi)加載器無(wú)法找到相應(yīng)的類(lèi),則拋出異常 
  19.             } 
  20.  
  21.             if (c == null) { 
  22.                 //當(dāng)父類(lèi)加載器無(wú)法加載時(shí),則調(diào)用findClass方法來(lái)加載該類(lèi) 
  23.                 long t1 = System.nanoTime(); 
  24.                 c = findClass(name); //用戶(hù)可通過(guò)覆寫(xiě)該方法,來(lái)自定義類(lèi)加載器 
  25.  
  26.                 //用于統(tǒng)計(jì)類(lèi)加載器相關(guān)的信息 
  27.                 sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0); 
  28.                 sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1); 
  29.                 sun.misc.PerfCounter.getFindClasses().increment(); 
  30.             } 
  31.         } 
  32.         if (resolve) { 
  33.             //對(duì)類(lèi)進(jìn)行l(wèi)ink操作 
  34.             resolveClass(c); 
  35.         } 
  36.         return c; 
  37.     } 

整個(gè)流程大致如下:

a.首先,檢查一下指定名稱(chēng)的類(lèi)是否已經(jīng)加載過(guò),如果加載過(guò)了,就不需要再加載,直接返回。

b.如果此類(lèi)沒(méi)有加載過(guò),那么,再判斷一下是否有父加載器;如果有父加載器,則由父加載器加載(即調(diào)用parent.loadClass(name, false);).或者是調(diào)用bootstrap類(lèi)加載器來(lái)加載。

c.如果父加載器及bootstrap類(lèi)加載器都沒(méi)有找到指定的類(lèi),那么調(diào)用當(dāng)前類(lèi)加載器的findClass方法來(lái)完成類(lèi)加載。


本文名稱(chēng):Java類(lèi)加載機(jī)制及類(lèi)加載器詳解
分享URL:http://www.dlmjj.cn/article/dpscjee.html