Java虛擬機類裝載:原理、完成與應用
發(fā)表時間:2024-05-22 來源:明輝站整理相關軟件相關文章人氣:
[摘要]一、引言 Java虛擬機(JVM)的類裝載就是指將包含在類文件中的字節(jié)碼裝載到JVM中, 并使其成為JVM一部分的過程。JVM的類動態(tài)裝載技術(shù)能夠在運行時刻動態(tài)地加載或者替換系統(tǒng)的某些功能模塊, 而不影響系統(tǒng)其他功能模塊的正常運行。本文將分析JVM中的類裝載系統(tǒng),探討JVM中類裝載的原理、實現(xiàn)...
一、引言 Java虛擬機(JVM)的類裝載就是指將包含在類文件中的字節(jié)碼裝載到JVM中, 并使其成為JVM一部分的過程。JVM的類動態(tài)裝載技術(shù)能夠在運行時刻動態(tài)地加載或者替換系統(tǒng)的某些功能模塊, 而不影響系統(tǒng)其他功能模塊的正常運行。本文將分析JVM中的類裝載系統(tǒng),探討JVM中類裝載的原理、實現(xiàn)以及應用。
二、Java虛擬機的類裝載實現(xiàn)與應用 2.1 裝載過程簡介
所謂裝載就是尋找一個類或是一個接口的二進制形式并用該二進制形式來構(gòu)造代表這個類或是這個接口的class對象的過程,其中類或接口的名稱是給定了的。當然名稱也可以通過計算得到,但是更常見的是通過搜索源代碼經(jīng)過編譯器編譯后所得到的二進制形式來構(gòu)造。
在Java中,類裝載器把一個類裝入Java虛擬機中,要經(jīng)過三個步驟來完成:裝載、鏈接和初始化,其中鏈接又可以分成校驗、準備和解析三步,除了解析外,其它步驟是嚴格按照順序完成的,各個步驟的主要工作如下:
裝載:查找和導入類或接口的二進制數(shù)據(jù);
鏈接:執(zhí)行下面的校驗、準備和解析步驟,其中解析步驟是可以選擇的;
校驗:檢查導入類或接口的二進制數(shù)據(jù)的正確性;
準備:給類的靜態(tài)變量分配并初始化存儲空間;
解析:將符號引用轉(zhuǎn)成直接引用;
初始化:激活類的靜態(tài)變量的初始化Java代碼和靜態(tài)Java代碼塊。
至于在類裝載和虛擬機啟動的過程中的具體細節(jié)和可能會拋出的錯誤,請參看《Java虛擬機規(guī)范》以及《深入Java虛擬機》,它們在網(wǎng)絡上面的資源地址是:
http://java.sun.com/docs/books/vmspec/2nd-edition/html/Preface.doc.html
http://www.artima.com/insidejvm/ed2/index.html
由于本文的討論重點不在此就不再多敘述。
2.2 裝載的實現(xiàn)
JVM中類的裝載是由ClassLoader和它的子類來實現(xiàn)的,Java ClassLoader 是一個重要的Java運行時系統(tǒng)組件。它負責在運行時查找和裝入類文件的類。
在Java中,ClassLoader是一個抽象類,它在包java.lang中,可以這樣說,只要了解了在ClassLoader中的一些重要的方法,再結(jié)合上面所介紹的JVM中類裝載的具體的過程,對動態(tài)裝載類這項技術(shù)就有了一個比較大概的掌握,這些重要的方法包括以下幾個:
、賚oadCass方法 loadClass(String name ,boolean resolve)其中name參數(shù)指定了JVM需要的類的名稱,該名稱以包表示法表示,如Java.lang.Object;resolve參數(shù)告訴方法是否需要解析類,在初始化類之前,應考慮類解析,并不是所有的類都需要解析,如果JVM只需要知道該類是否存在或找出該類的超類,那么就不需要解析。這個方法是ClassLoader 的入口點。
②defineClass方法 這個方法接受類文件的字節(jié)數(shù)組并把它轉(zhuǎn)換成Class對象。字節(jié)數(shù)組可以是從本地文件系統(tǒng)或網(wǎng)絡裝入的數(shù)據(jù)。它把字節(jié)碼分析成運行時數(shù)據(jù)結(jié)構(gòu)、校驗有效性等等。
③findSystemClass方法 findSystemClass方法從本地文件系統(tǒng)裝入文件。它在本地文件系統(tǒng)中尋找類文件,如果存在,就使用defineClass將字節(jié)數(shù)組轉(zhuǎn)換成Class對象,以將該文件轉(zhuǎn)換成類。當運行Java應用程序時,這是JVM 正常裝入類的缺省機制。
、躵esolveClass方法 resolveClass(Class c)方法解析裝入的類,如果該類已經(jīng)被解析過那么將不做處理。當調(diào)用loadClass方法時,通過它的resolve 參數(shù)決定是否要進行解析。
、輋indLoadedClass方法 當調(diào)用loadClass方法裝入類時,調(diào)用findLoadedClass 方法來查看ClassLoader是否已裝入這個類,如果已裝入,那么返回Class對象,否則返回NULL。如果強行裝載已存在的類,將會拋出鏈接錯誤。
2.3 裝載的應用
一般來說,我們使用虛擬機的類裝載時需要繼承抽象類java.lang.ClassLoader,其中必須實現(xiàn)的方法是loadClass(),對于這個方法需要實現(xiàn)如下操作:(1) 確認類的名稱;(2) 檢查請求要裝載的類是否已經(jīng)被裝載;(3) 檢查請求加載的類是否是系統(tǒng)類;(4) 嘗試從類裝載器的存儲區(qū)獲取所請求的類;(5) 在虛擬機中定義所請求的類;(6) 解析所請求的類;(7) 返回所請求的類。
所有的Java 虛擬機都包括一個內(nèi)置的類裝載器,這個內(nèi)置的類庫裝載器被稱為根裝載器(bootstrap ClassLoader)。根裝載器的特殊之處是它只能夠裝載在設計時刻已知的類,因此虛擬機假定由根裝載器所裝載的類都是安全的、可信任的,可以不經(jīng)過安全認證而直接運行。當應用程序需要加載并不是設計時就知道的類時,必須使用用戶自定義的裝載器(user-defined ClassLoader)。下面我們舉例說明它的應用。
public abstract class MultiClassLoader extends ClassLoader{
...
public synchronized Class loadClass(String s, boolean flag)
throws ClassNotFoundException
{
/* 檢查類s是否已經(jīng)在本地內(nèi)存*/
Class class1 = (Class)classes.get(s);
/* 類s已經(jīng)在本地內(nèi)存*/
if(class1 != null) return class1;
try/*用默認的ClassLoader 裝入類*/ {
class1 = super.findSystemClass(s);
return class1;
}
catch(ClassNotFoundException _ex) {
System.out.println(">> Not a system class.");
}
/* 取得類s的字節(jié)數(shù)組*/
byte abyte0[] = loadClassBytes(s);
if(abyte0 == null) throw new ClassNotFoundException();
/* 將類字節(jié)數(shù)組轉(zhuǎn)換為類*/
class1 = defineClass(null, abyte0, 0, abyte0.length);
if(class1 == null) throw new ClassFormatError();
if(flag) resolveClass(class1); /*解析類*/
/* 將新加載的類放入本地內(nèi)存*/
classes.put(s, class1);
System.out.println(">> Returning newly loaded class.");
/* 返回已裝載、解析的類*/
return class1;
}
...
}