<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    posts - 188,comments - 176,trackbacks - 0

    一、引言

    Java虛擬機(JVM)的類裝載就是指將包含在類文件中的字節碼裝載到JVM中, 并使其成為JVM一部分的過程。JVM的類動態裝載技術能夠在運行時刻動態地加載或者替換系統的某些功能模塊, 而不影響系統其他功能模塊的正常運行。本文將分析JVM中的類裝載系統,探討JVM中類裝載的原理、實現以及應用。

     

    二、Java虛擬機的類裝載實現與應用

     

    2.1  裝載過程簡介

     

    所謂裝載就是尋找一個類或是一個接口的二進制形式并用該二進制形式來構造代表這個類或是這個接口的class對象的過程,其中類或接口的名稱是給定了的。當然名稱也可以通過計算得到,但是更常見的是通過搜索源代碼經過編譯器編譯后所得到的二進制形式來構造。

     

    在Java中,類裝載器把一個類裝入Java虛擬機中,要經過三個步驟來完成:裝載、鏈接和初始化,其中鏈接又可以分成校驗、準備和解析三步,除了解析外,其它步驟是嚴格按照順序完成的,各個步驟的主要工作如下:

     

    • 裝載:查找和導入類或接口的二進制數據;
    • 鏈接:執行下面的校驗、準備和解析步驟,其中解析步驟是可以選擇的;
    • 校驗:檢查導入類或接口的二進制數據的正確性;
    • 準備:給類的靜態變量分配并初始化存儲空間;
    • 解析:將符號引用轉成直接引用;
    • 初始化:激活類的靜態變量的初始化Java代碼和靜態Java代碼塊。

     

     

    至于在類裝載和虛擬機啟動的過程中的具體細節和可能會拋出的錯誤,請參看《Java虛擬機規范》以及《深入Java虛擬機》,它們在網絡上面的資源地址是:

    http://java.sun.com/docs/books/vmspec/2nd-edition/html/Preface.doc.htmlhttp://www.artima.com/insidejvm/ed2/index.html

     

    2.2  裝載的實現

     

    JVM中類的裝載是由ClassLoader和它的子類來實現的,Java ClassLoader 是一個重要的Java運行時系統組件。它負責在運行時查找和裝入類文件的類。

     

    在Java中,ClassLoader是一個抽象類,它在包java.lang中,可以這樣說,只要了解了在ClassLoader中的一些重要的方法,再結合上面所介紹的JVM中類裝載的具體的過程,對動態裝載類這項技術就有了一個比較大概的掌握,這些重要的方法包括以下幾個:

     

    ①loadCass方法  loadClass(String name ,boolean resolve)其中name參數指定了JVM需要的類的名稱,該名稱以包表示法表示,如Java.lang.Object;resolve參數告訴方法是否需要解析類,在初始化類之前,應考慮類解析,并不是所有的類都需要解析,如果JVM只需要知道該類是否存在或找出該類的超類,那么就不需要解析。這個方法是ClassLoader 的入口點。

     

    ②defineClass方法  這個方法接受類文件的字節數組并把它轉換成Class對象。字節數組可以是從本地文件系統或網絡裝入的數據。它把字節碼分析成運行時數據結構、校驗有效性等等。

     

    ③findSystemClass方法  findSystemClass方法從本地文件系統裝入文件。它在本地文件系統中尋找類文件,如果存在,就使用defineClass將字節數組轉換成Class對象,以將該文件轉換成類。當運行Java應用程序時,這是JVM 正常裝入類的缺省機制。

     

    ④resolveClass方法  resolveClass(Class c)方法解析裝入的類,如果該類已經被解析過那么將不做處理。當調用loadClass方法時,通過它的resolve 參數決定是否要進行解析。

     

    ⑤findLoadedClass方法  當調用loadClass方法裝入類時,調用findLoadedClass 方法來查看ClassLoader是否已裝入這個類,如果已裝入,那么返回Class對象,否則返回NULL。如果強行裝載已存在的類,將會拋出鏈接錯誤。

     

    2.3  裝載的應用

     

    一般來說,我們使用虛擬機的類裝載時需要繼承抽象類java.lang.ClassLoader,其中必須實現的方法是loadClass(),對于這個方法需要實現如下操作:(1) 確認類的名稱;(2) 檢查請求要裝載的類是否已經被裝載;(3) 檢查請求加載的類是否是系統類;(4) 嘗試從類裝載器的存儲區獲取所請求的類;(5) 在虛擬機中定義所請求的類;(6) 解析所請求的類;(7) 返回所請求的類。

     

    所有的Java 虛擬機都包括一個內置的類裝載器,這個內置的類庫裝載器被稱為根裝載器(bootstrap ClassLoader)。根裝載器的特殊之處是它只能夠裝載在設計時刻已知的類,因此虛擬機假定由根裝載器所裝載的類都是安全的、可信任的,可以不經過安全認證而直接運行。當應用程序需要加載并不是設計時就知道的類時,必須使用用戶自定義的裝載器(user-defined ClassLoader)。

     

    下面我們舉例說明它的應用。

     
    

    public abstract class MultiClassLoaderextends ClassLoader{

       ...

       public synchronized Class loadClass(String s, booleanflag)

           throws ClassNotFoundException

       {

           /* 已經在本地內存*/

           if(class1 != null)       try/*返回已裝載、解析的類*/

           return class1;

       }

       ...

    }

     

     

    三、Java虛擬機的類裝載原理

     

    前面我們已經知道,一個Java應用程序使用兩種類型的類裝載器:根裝載器(bootstrap)和用戶定義的裝

    載器(user-defined)。根裝載器是Java虛擬機實現的一部分,舉個例子來說,如果一個Java虛擬機是在

    現在已經存在并且正在被使用的操作系統的頂部用C程序來實現的,那么根裝載器將是那些C程序的一部

    分。根裝載器以某種默認的方式將類裝入,包括那些Java API的類。在運行期間一個Java程序能安裝用

    戶自己定義的類裝載器。根裝載器是虛擬機固有的一部分,而用戶定義的類裝載器則不是,它是用Java

    語言寫的,被編譯成class文件之后然后再被裝入到虛擬機,并像其它的任何對象一樣可以被實例化。

    Java類裝載器的體系結構如下所示:

     

     

     

     

     

     

     

    圖1  Java的類裝載的體系結構

     

     

     

    Java的類裝載模型是一種代理(delegation)模型。當JVM 要求類裝載器CL(ClassLoader)裝載一個類

    時,CL首先將這個類裝載請求轉發給他的父裝載器。只有當父裝載器沒有裝載并無法裝載這個類時,CL才

    獲得裝載這個類的機會。這樣, 所有類裝載器的代理關系構成了一種樹狀的關系。樹的根是類的根裝載

    器(bootstrap ClassLoader) , 在JVM 中它以"null"表示。除根裝載器以外的類裝載器有且僅有一個父

    裝載器。在創建一個裝載器時, 如果沒有顯式地給出父裝載器, 那么JVM將默認系統裝載器為其父裝載器

    。Java的基本類裝載器代理結構如圖2所示:

     

     

     

     

     

    圖2  Java類裝載的代理結構

     

     

     

    下面針對各種類裝載器分別進行詳細的說明。

     

     

    根(Bootstrap) 裝載器:該裝載器沒有父裝載器,它是JVM實現的一部分,從sun.boot.class.path裝載運

    行時庫的核心代碼。

     

    擴展(Extension) 裝載器:繼承的父裝載器為根裝載器,不像根裝載器可能與運行時的操作系統有關,這

    個類裝載器是用純Java代碼實現的,它從java.ext.dirs (擴展目錄)中裝載代碼。

     

    系統(System or Application) 裝載器:裝載器為擴展裝載器,我們都知道在安裝JDK的時候要設置環境

    變量(CLASSPATH ),這個類裝載器就是從java.class.path(CLASSPATH 環境變量)中裝載代碼的,它也是

    用純Java代碼實現的,同時還是用戶自定義類裝載器的缺省父裝載器。

     

    小應用程序(Applet) 裝載器: 裝載器為系統裝載器,它從用戶指定的網絡上的特定目錄裝載小應用程序

    代碼。

     

     

    在設計一個類裝載器的時候,應該滿足以下兩個條件:

     

    對于相同的類名,類裝載器所返回的對象應該是同一個類對象


    如果類裝載器CL1將裝載類C的請求轉給類裝載器CL2,那么對于以下的類或接口,CL1和CL2應該返回同一

    個類對象:a)S為C的直接超類;b)S為C的直接超接口;c)S為C的成員變量的類型;d)S為C的成員方法或構建

    器的參數類型;e)S為C的成員方法的返回類型。


    每個已經裝載到JVM中的類都隱式含有裝載它的類裝載器的信息。類方法getClassLoader 可以得到裝載

    這個類的類裝載器。一個類裝載器認識的類包括它的父裝載器認識的類和它自己裝載的類,可見類裝載

    器認識的類是它自己裝載的類的超集。注意我們可以得到類裝載器的有關的信息,但是已經裝載到JVM中

    的類是不能更改它的類裝載器的。

     

    Java中的類的裝載過程也就是代理裝載的過程。比如:Web瀏覽器中的JVM需要裝載一個小應用程序

    TestApplet。JVM調用小應用程序裝載器ACL(Applet ClassLoader)來完成裝載。ACL首先請求它的父裝載

    器, 即系統裝載器裝載TestApplet是否裝載了這個類, 由于TestApplet不在系統裝載器的裝載路徑中,

    所以系統裝載器沒有找到這個類, 也就沒有裝載成功。接著ACL自己裝載TestApplet。ACL通過網絡成功

    地找到了TestApplet.class 文件并將它導入到了JVM中。在裝載過程中, JVM發現TestAppet是從超類

    java.applet.Applet繼承的。所以JVM再次調用ACL來裝載java.applet.Applet類。ACL又再次按上面的順

    序裝載Applet類, 結果ACL發現他的父裝載器已經裝載了這個類, 所以ACL就直接將這個已經裝載的類返

    回給了JVM , 完成了Applet類的裝載。接下來,Applet類的超類也一樣處理。最后, TestApplet及所有有

    關的類都裝載到了JVM中。

     

     

    四、結論

     

    類的動態裝載機制是JVM的一項核心技術, 也是容易被忽視而引起很多誤解的地方。本文介紹了JVM中類

    裝載的原理、實現以及應用,尤其分析了ClassLoader的結構、用途以及如何利用自定義的ClassLoader

    裝載并執行Java類,希望能使讀者對JVM中的類裝載有一個比較深入的理解。

     

     

    轉:http://blog.csdn.net/bon_jovi/archive/2004/08/11/71283.aspx

     

     

    posted on 2007-05-24 10:48 cheng 閱讀(264) 評論(0)  編輯  收藏 所屬分類: JBS
    主站蜘蛛池模板: 深夜福利在线视频免费| 久久精品国产亚洲av日韩| 免费国产一级特黄久久| 免费无码精品黄AV电影| 一二三四视频在线观看中文版免费| 国产成人精品久久免费动漫| 1000部啪啪毛片免费看| 久久久久久精品成人免费图片| 3d动漫精品啪啪一区二区免费| 最近免费mv在线电影| 91成人免费观看网站| A级毛片内射免费视频| 免费无码又爽又刺激高潮| 国产精品色午夜视频免费看| 四虎影视在线永久免费观看| 亚洲国产天堂久久综合| 黑人大战亚洲人精品一区| 亚洲精品国产精品乱码不99| 亚洲精品高清国产一久久| 亚洲理论片在线观看| 亚洲人成未满十八禁网站| 日韩色视频一区二区三区亚洲| 无码免费又爽又高潮喷水的视频| 久久久精品国产亚洲成人满18免费网站| 99re8这里有精品热视频免费| 国产一级婬片A视频免费观看| 日韩内射激情视频在线播放免费| 91精品成人免费国产片| 免费视频淫片aa毛片| 亚洲国产精品成人网址天堂 | 一区二区三区视频免费| 日韩a级无码免费视频| 亚洲免费视频播放| 日韩成人在线免费视频| 国产成人高清亚洲| 久久久亚洲欧洲日产国码是AV| 亚洲经典千人经典日产| 国产精品美女久久久免费| 99精品视频在线免费观看| 成人一a毛片免费视频| 精品国产日韩亚洲一区|