溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Java高級(jí)之虛擬機(jī)加載機(jī)制的示例分析

發(fā)布時(shí)間:2021-07-23 11:00:42 來源:億速云 閱讀:127 作者:小新 欄目:編程語言

這篇文章給大家分享的是有關(guān)Java高級(jí)之虛擬機(jī)加載機(jī)制的示例分析的內(nèi)容。小編覺得挺實(shí)用的,因此分享給大家做個(gè)參考,一起跟隨小編過來看看吧。

Jvm要加載的是二進(jìn)制流,可以是.class文件形式,也可以是其他形式,按照它加載的標(biāo)準(zhǔn)來設(shè)計(jì)就不會(huì)有太大問題。

以下主要就機(jī)制和標(biāo)準(zhǔn)兩個(gè)問題分析一番:

首先來Java類文件的加載機(jī)制 ,跟變量的加載機(jī)制類似,它先把Class文件加載入內(nèi)存,再對(duì)數(shù)據(jù)進(jìn)行驗(yàn)證、解析和初始化,最終形成虛擬機(jī)可以直接使用的Java類型。由于Java是采用JIT機(jī)制,所以加載時(shí)會(huì)比較慢,但優(yōu)點(diǎn)也明顯,具有高度靈活性,支持動(dòng)態(tài)加載和動(dòng)態(tài)連接。

接下來就講講類的加載過程:

一個(gè)類加載的基本過程是按照下面的順序 來,但也有不嚴(yán)格按照這個(gè)順序來的,也有打亂順序來的,如動(dòng)態(tài)加載就得先初始化再解析。

1、加載

由虛擬機(jī)自行決定,但也有由于下面的階段要執(zhí)行而執(zhí)行上面階段的情況。

這時(shí)虛擬機(jī)會(huì)做三件事:

第一、通過全限定名讀取文件的二進(jìn)制流;

第二、把文件里的靜態(tài)方法和變量放到方法區(qū)中;

第三、生成一個(gè)對(duì)象放入堆中,作為訪問入口。

注意第一條,僅是讀取二進(jìn)制流,沒說具體從什么文件中讀,也沒說從哪里讀,所以造就Java很強(qiáng)的擴(kuò)展性,可以從Jar、Zip中,也可以從網(wǎng)絡(luò)層、數(shù)據(jù)庫層等 。

主要是對(duì)象和方法區(qū)的聲明。

2、驗(yàn)證

確保二進(jìn)制流符合虛擬機(jī)的要求, 不符合會(huì)報(bào)VerifyError。

第一、文件格式驗(yàn)證,是否有魔數(shù),是否符合Java文件的要求;

第二、元數(shù)據(jù)驗(yàn)證,是否符合Java代碼規(guī)范,如abstract類是否直接被實(shí)例化,普通類有無間接或直接父類Object等;第三、字節(jié)碼驗(yàn)證,對(duì)數(shù)據(jù)流和控制流進(jìn)行分析,保證不會(huì)做出危害虛擬機(jī)的行為,如 是否調(diào)用不存在的指令,是否把父類賦值給子類,是否把對(duì)象賦值給一個(gè)非此類型的對(duì)象等;

第四、符號(hào)引用驗(yàn)證,主要是類、變量、方法描述是否能找的到,如全限定名是否能找到該文件,是否具有可訪問性等。

主要對(duì)內(nèi)部結(jié)構(gòu)的判定

3、準(zhǔn)備

為類變量賦初值,通常為0值如靜態(tài)變量,而不會(huì)為實(shí)例變量賦值。

4、解析

將常量池中的符號(hào)引用轉(zhuǎn)化為直接引用的過程。這里說的符號(hào)引用指變量類型,直接引用指可以直接定位到對(duì)象的句柄。類、方法、字段、接口解析,根據(jù)全限定名獲得相關(guān)對(duì)象,拿到它的類型,若無對(duì)所在類訪問權(quán)會(huì)拋出IllegalAccessError,無字段NoSuchFieldError,無方法NoSuchMethodError,是類不是接口會(huì)拋出IncompatibleClassChangeError

5、初始化

根據(jù)程序要求加載類和必要的資源。有且僅有四種情況,需要主動(dòng)初始化后才能執(zhí)行接下來的操作 ,所以要先執(zhí)行上面的四步。

第一、有new或static關(guān)鍵字的類,new生成對(duì)象,static靜態(tài)加載,這兩個(gè)很明顯要執(zhí)行初始化了;

第二、使用類有父類,這沒辦法了;

第三、反射類里的方法,那肯定要初始化了對(duì)不對(duì);

第四、執(zhí)行的主類,用main方法的類。其他被動(dòng)初始化的情況不需要考慮。

小例子:

public class SuperClass {
static {
System.out.println(“SuperClass!!”);
}
public static int value = 1;
}
public class SubClass extends SuperClass {
static {
System.out.println(“SubClass!!”);
}
}
public class TestClassLoad {
public static void main(String[] args) {
System.out.println(SubClass.value);
SuperClass superClass = new SubClass();
}
}
SuperClass!!
1
SubClass!!

執(zhí)行結(jié)果說明一個(gè)問題: 子類調(diào)用父類變量的時(shí)候 ,子類沒有初始化,因?yàn)?此時(shí)的代碼關(guān)系跟子類無關(guān) ;子類初始化的時(shí)候,父類也沒有再初始化,因?yàn)?父類在當(dāng)前方法體中已經(jīng)初始化 過了。接口與父類的唯一區(qū)別在于, 接口初始化不會(huì)要求父接口,只有用到父接口才會(huì)初始化 ,同樣的都會(huì)生成類構(gòu)造器。

這個(gè)時(shí)候加載類構(gòu)造器,會(huì)初始化類中所有變量,當(dāng)然父類先于子類初始化

6、使用

加載完之后,該怎么樣調(diào)用怎么樣調(diào)用,繪圖啊,計(jì)算啊等等

7、卸載

類不再被調(diào)用

兩個(gè)類是否相等,主要在于第一使用同一個(gè)加載器加載,第二全限定名地址一致

為什么要提出上面的問題呢?接下來要講講虛擬機(jī)的一個(gè)加載機(jī)制。

在java虛擬機(jī)的角度來看,有兩種類加載器,一種叫系統(tǒng)加載器(Bootstrap ClassLoader),一種叫自定義加載器(extends ClassLoader),這種呢又分為兩個(gè),一種叫應(yīng)用加載器,一種叫擴(kuò)展類加載器,一般默認(rèn)為前者;而我們的應(yīng)用程序加載主要由上面三個(gè)加載器相互配合完成的。三者的關(guān)系如Application–>Extension–>Bootsrap,雙親委派機(jī)制是指兩兩以組合的方式,子加載器先去調(diào)用父加載器的方法,沒找到目標(biāo)對(duì)象再去用子加載器

偽代碼如下:

loadClass(String name,boolean resolve){
Class c=findLoadedClass()
if(c==null){
if(parent !=null)
c=parent.loadClass(name,false);
c=findBootstrapClassOrNull(name);
}catch(ClassNotFoundException e){ }
if(c==null)
c=findClass(name);
}

Java提倡我們?nèi)グ炎约赫{(diào)用類的邏輯寫在findClass里,這樣有助于雙親委派機(jī)制的正常使用。

破壞1、重寫loadClass

破壞2、使用線程上下文加載器去讓父加載器去調(diào)用子加載器的方法

破壞3、熱加載 現(xiàn)在常用的做法是 自定義類加載器并 將原bug模塊覆蓋-OSGI

但由于自定義加載器之間的規(guī)則如果混亂,出現(xiàn)同時(shí)互相引用的問題,那么會(huì)最終找不到類,而出現(xiàn)線程死鎖和內(nèi)存泄露的問題。

關(guān)于熱修復(fù),也被稱為插件,目前比較流行的有HotFix、Nuwa、DroidFix、AndFix等,這些框架均可以在github或其他地方找到,原理如上,方法多樣,有覆蓋的、有重定向的等等,通過配置、設(shè)置action等方式;而作為插件需要滿足以下條件:

1、可以獨(dú)立安裝,但不可獨(dú)立運(yùn)行

2、具有向下兼容性,即可拓展性

3、只能運(yùn)行在宿主程序中,而且可以被禁用、替換

感謝各位的閱讀!關(guān)于“Java高級(jí)之虛擬機(jī)加載機(jī)制的示例分析”這篇文章就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,讓大家可以學(xué)到更多知識(shí),如果覺得文章不錯(cuò),可以把它分享出去讓更多的人看到吧!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI