偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

Java類加載機制及類加載器詳解

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

[[409199]]

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

一、類加載機制

1.什么是類加載?

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

2.類加載的過程

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

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

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

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

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

②.使用java.lang.reflect包的方法對類進行反射調(diào)用的時候。

③.當(dāng)初始化一個類的時候,發(fā)現(xiàn)其父類還沒有進行過初始化,則需要先出發(fā)父類的初始化。

④.當(dāng)虛擬機啟動時,用戶需要指定一個要執(zhí)行的主類(包含main()方法的那個類),虛擬機會先初始化這個主類。

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

4.類加載的具體過程

加載:

①.通過一個類的全限定名來獲取定義此類的二進制字節(jié)流

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

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

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

包含四個階段的校驗動作

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

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

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

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

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

初始化:

類加載過程中的最后一步。

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

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

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

簡單地說,初始化就是對類變量進行賦值及執(zhí)行靜態(tài)代碼塊。

二、類加載器

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

1.類加載器分類:

不同的類加載器負責(zé)加載不同的類。主要分為兩類。

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

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

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

其他類加載器:由Java語言實現(xiàn),繼承自抽象類ClassLoader。

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

2.雙親委派模型

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

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

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

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

  1. protected Class<?> loadClass(String name, boolean resolve) 
  2.     throws ClassNotFoundException 
  3.     synchronized (getClassLoadingLock(name)) { 
  4.         //檢查該類是否已經(jīng)加載過 
  5.         Class c = findLoadedClass(name); 
  6.         if (c == null) { 
  7.             //如果該類沒有加載,則進入該分支 
  8.             long t0 = System.nanoTime(); 
  9.             try { 
  10.                 if (parent != null) { 
  11.                     //當(dāng)父類的加載器不為空,則通過父類的loadClass來加載該類 
  12.                     c = parent.loadClass(namefalse); 
  13.                 } else { 
  14.                     //當(dāng)父類的加載器為空,則調(diào)用啟動類加載器來加載該類 
  15.                     c = findBootstrapClassOrNull(name); 
  16.                 } 
  17.             } catch (ClassNotFoundException e) { 
  18.                 //非空父類的類加載器無法找到相應(yīng)的類,則拋出異常 
  19.             } 
  20.  
  21.             if (c == null) { 
  22.                 //當(dāng)父類加載器無法加載時,則調(diào)用findClass方法來加載該類 
  23.                 long t1 = System.nanoTime(); 
  24.                 c = findClass(name); //用戶可通過覆寫該方法,來自定義類加載器 
  25.  
  26.                 //用于統(tǒng)計類加載器相關(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.             //對類進行l(wèi)ink操作 
  34.             resolveClass(c); 
  35.         } 
  36.         return c; 
  37.     } 

整個流程大致如下:

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

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

 

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

 

責(zé)任編輯:武曉燕 來源: 愛寫B(tài)ug的麥洛
相關(guān)推薦

2020-10-26 11:20:04

jvm類加載Java

2019-07-24 08:34:35

Java對象數(shù)據(jù)結(jié)構(gòu)

2009-02-03 09:42:53

JAVA類JVM指令forName方法

2024-03-12 07:44:53

JVM雙親委托機制類加載器

2012-02-09 10:31:17

Java

2025-07-01 07:41:37

Java類加載器雙親委派

2023-10-31 16:00:51

類加載機制Java

2024-04-09 08:41:41

JVM類加載Java

2021-01-06 09:01:05

javaclass

2024-09-06 09:37:45

WebApp類加載器Web 應(yīng)用

2023-08-02 08:38:27

JVM加載機制

2017-03-08 10:30:43

JVMJava加載機制

2023-05-10 11:07:18

2017-09-20 08:07:32

java加載機制

2025-06-26 03:33:00

2024-09-04 09:47:21

2011-02-25 09:23:00

Java類加載器

2024-12-02 09:01:23

Java虛擬機內(nèi)存

2024-12-04 09:01:55

引導(dǎo)類加載器C++

2021-04-29 11:18:14

JVM加載機制
點贊
收藏

51CTO技術(shù)棧公眾號