自定義類加載器

重要概念

  1. 通過自定義類加載器镣典,程序就可以加載在編譯時并不知道在哪或者尚未存在的類或者接口(.class文件);
  2. 類加載器加載的是.class文件兔港,所以加載之前需要.class文件存在武契;

自定義類加載器的步驟

  1. 繼承自ClassLoader
  2. override父類ClassLoader的findClass()方法伦意;
    查看ClassLoader的源碼就能很好的理解為什么類加載器的步驟是這樣.

執(zhí)行原理

  • 創(chuàng)建自定義類加載器對象時,默認(rèn)(或者顯示指定)其父類加載器硼补;
  • 顯示調(diào)用繼承來的loadClass()方法來加載類驮肉,此時在loadClass()方法內(nèi)部將執(zhí)行雙親委派的邏輯,如果父類加載沒能成功加載類已骇,則調(diào)用override父類的findClass()方法來實現(xiàn)自定義類加載器加載類的邏輯:
    • 將要加載的類的.class文件中的內(nèi)容輸入到byte[]數(shù)組中离钝;
    • 調(diào)用父類(ClassLoader)的defineClass()方法完成真正的類加載;

示例

定義一個將要被自定義類加載器加載的類

package com.univ;//注意這里Sample類的全路徑名為com.univ.Sample褪储,將供defineClass()使用
public class Sample{

}

自定義類加載器

class MyClassLoader extends ClassLoader {//1.繼承ClassLoader
 
    public MyClassLoader() {
        super();//此時調(diào)用父類ClassLoader中的構(gòu)造函數(shù)卵渴,默認(rèn)將AppClassLoader作為父類加載器
    }
 
    public MyClassLoader(ClassLoader parent) {
        super(parent);//顯示指定此類加載器的父類加載器
    }
 
    @Override
    protected Class<?> findClass(String name) throws ClassNotFoundException {//2.覆寫findClass()方法 
        byte[] data = null; 
        FileInputStream fis = null; 
        try {
            fis = new FileInputStream("d:\\myclass\\Sample.class");//假設(shè)編譯后的Sample.class文件放在d盤的myclass目錄下
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        }
        ByteArrayOutputStream abos = new ByteArrayOutputStream();
        int ch = 0;
        try {
            while (-1 != (ch = fis.read())) {
                abos.write(ch);
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
        data = abos.toByteArray();   //把文件輸出流中的字節(jié)轉(zhuǎn)換成字節(jié)數(shù)組 
        return this.defineClass(name,data, 0, data.length); 
    } 
    //這里將字節(jié)文件內(nèi)容讀入到字節(jié)數(shù)組中有更簡單的方法
    // byte[] data = new byte[fis.avaiable()];fis.read(b);
    //defineClass(byte[] b, int off, int len)已經(jīng)被廢棄,用下面的方法取代:
    // defineClass(String name, byte[] b, int off, int len),其中name是要加載的類的全路徑名(加載的雖然是.class文件鲤竹,但這里的全路徑名指的是.java文件的全路徑名+類名)
}

測試代碼

@Test
public void test1() {
    MyClassLoader myClassLoader = new MyClassLoader();
    try {
        Class clazz = myClassLoader.loadClass("com.univ.Sample");
        System.out.println("加載Sample的類加載器為: " + clazz.getClassLoader());
    } catch (ClassNotFoundException e) {
        e.printStackTrace();
    }
}

注意:此時類路徑下不能有Sample.java浪读,否則將被AppClassLoader加載到

ClassLoader中重要方法解析

public Class<?> loadClass(String name) throws ClassNotFoundException {
        return loadClass(name, false);
}

封裝了雙親委派模式的邏輯:先交由父類加載,父類沒有加載到再由自己加載.所以自定義的類加載器不要override此方法

 protected Class<?> findClass(String name) throws ClassNotFoundException {
        throw new ClassNotFoundException(name);
}

自己加載類的邏輯:也就是當(dāng)父類沒有加載到時自己如何加載辛藻,自然自定義的類加載器中需要(也只需要)復(fù)寫此方法

protected final Class<?> defineClass(String name, byte[] b, int off, int len) throws ClassFormatError
{ 

}

將由自定義的findClass()方法調(diào)用(見示例)碘橘,findClass()的邏輯就是利用流將.class文件中的內(nèi)容轉(zhuǎn)變成byte[],然后調(diào)用父類的defineClass(...)真正加載類吱肌。
注意蛹屿,這里的name是要加載的類的全路徑名,如這里的com.univ.Sample(不是.class文件所在的目錄岩榆,也不是Sample).
2016/3/20 21:58:41


補(bǔ)充:類加載器繼承關(guān)系

ClassLoader類定義如下:

public abstract class ClassLoader { 
   private final ClassLoader parent;
 
   public Class<?> loadClass(String name) throws ClassNotFoundException {
        return loadClass(name, false);
   }
 
   protected Class<?> findClass(String name) throws ClassNotFoundException {
        throw new ClassNotFoundException(name);
   } 
}

注意

  1. 其中的loadClass方法封裝了雙親委派的邏輯错负,findClass專門用來繼承,在這里寫自定義類加載器的加載類的邏輯勇边;
  2. 雖然ExtClassLoader是AppClassLoader的父類加載器犹撒,但AppClassLoader并不繼承自ExtClassLoader,因為類加載器之間的關(guān)系不是通過繼承完成粒褒,而是通過組合完成的(見上面的parent屬性)只泼;
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末手形,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌弹沽,老刑警劉巖,帶你破解...
    沈念sama閱讀 212,816評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件盗痒,死亡現(xiàn)場離奇詭異钩杰,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)苛萎,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,729評論 3 385
  • 文/潘曉璐 我一進(jìn)店門桨昙,熙熙樓的掌柜王于貴愁眉苦臉地迎上來检号,“玉大人,你說我怎么就攤上這事蛙酪∑肟粒” “怎么了?”我有些...
    開封第一講書人閱讀 158,300評論 0 348
  • 文/不壞的土叔 我叫張陵桂塞,是天一觀的道長凹蜂。 經(jīng)常有香客問我,道長阁危,這世上最難降的妖魔是什么玛痊? 我笑而不...
    開封第一講書人閱讀 56,780評論 1 285
  • 正文 為了忘掉前任,我火速辦了婚禮欲芹,結(jié)果婚禮上卿啡,老公的妹妹穿的比我還像新娘。我一直安慰自己菱父,他們只是感情好颈娜,可當(dāng)我...
    茶點故事閱讀 65,890評論 6 385
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著浙宜,像睡著了一般官辽。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上粟瞬,一...
    開封第一講書人閱讀 50,084評論 1 291
  • 那天同仆,我揣著相機(jī)與錄音,去河邊找鬼裙品。 笑死俗批,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的市怎。 我是一名探鬼主播岁忘,決...
    沈念sama閱讀 39,151評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼区匠!你這毒婦竟也來了干像?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,912評論 0 268
  • 序言:老撾萬榮一對情侶失蹤驰弄,失蹤者是張志新(化名)和其女友劉穎麻汰,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體戚篙,經(jīng)...
    沈念sama閱讀 44,355評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡五鲫,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,666評論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了已球。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片臣镣。...
    茶點故事閱讀 38,809評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡辅愿,死狀恐怖智亮,靈堂內(nèi)的尸體忽然破棺而出忆某,到底是詐尸還是另有隱情,我是刑警寧澤阔蛉,帶...
    沈念sama閱讀 34,504評論 4 334
  • 正文 年R本政府宣布弃舒,位于F島的核電站,受9級特大地震影響状原,放射性物質(zhì)發(fā)生泄漏聋呢。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 40,150評論 3 317
  • 文/蒙蒙 一颠区、第九天 我趴在偏房一處隱蔽的房頂上張望削锰。 院中可真熱鬧,春花似錦毕莱、人聲如沸器贩。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,882評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽蛹稍。三九已至,卻和暖如春部服,著一層夾襖步出監(jiān)牢的瞬間唆姐,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,121評論 1 267
  • 我被黑心中介騙來泰國打工廓八, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留奉芦,地道東北人。 一個月前我還...
    沈念sama閱讀 46,628評論 2 362
  • 正文 我出身青樓剧蹂,卻偏偏與公主長得像声功,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子国夜,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 43,724評論 2 351

推薦閱讀更多精彩內(nèi)容