離弦之Ray

            BlogJava :: 首頁(yè) :: 聯(lián)系 :: 聚合  :: 管理
            55 Posts :: 0 Stories :: 33 Comments :: 0 Trackbacks
              Java中一共有四個(gè)類(lèi)加載器,之所以叫類(lèi)加載器,是程序要用到某個(gè)類(lèi)的時(shí)候,要用類(lèi)加載器載入內(nèi)存。
              這四個(gè)類(lèi)加載器分別為:Bootstrap ClassLoaderExtension ClassLoaderAppClassLoader
          URLClassLoader,他們的作用其實(shí)從名字就可以大概推測(cè)出來(lái)了。其中AppClassLoader在很多地方被叫做System ClassLoader

          Bootstrap ClassLoader是在JVM開(kāi)始運(yùn)行的時(shí)候加載java的核心類(lèi),是用C++編寫(xiě)的,它用來(lái)加載核心類(lèi)庫(kù),在JVM源代碼中這樣寫(xiě)道:
          static const char classpathFormat[] =
          "%/lib/rt.jar:"
          "%/lib/i18n.jar:"
          "%/lib/sunrsasign.jar:"
          "%/lib/jsse.jar:"
          "%/lib/jce.jar:"
          "%/lib/charsets.jar:"
          "%/classes";
          Extension ClassLoader是用來(lái)加載擴(kuò)展類(lèi),即/lib/ext中的類(lèi)。
          AppClassLoader用來(lái)加載Classpath的類(lèi),是和我們關(guān)系最密切的類(lèi)。
          URLClassLoader用來(lái)加載網(wǎng)絡(luò)上遠(yuǎn)程的類(lèi),暫且不討論。

          它們之間的關(guān)系:

          1.Parent-Child,按順序從大到小。不是簡(jiǎn)單的繼承關(guān)系。

          2.ClassLoader有個(gè)getParent的方法,但是Ext ClassLoader調(diào)用后得到的是null,bootstrap是JVM自己的,用戶看不到。

          3.classloader的委托機(jī)制:當(dāng)?shù)燃?jí)比較低的ClassLoader要加載某個(gè)類(lèi)的時(shí)候,它首先會(huì)請(qǐng)求Parent加載器來(lái)加載,Parent再請(qǐng)求它的Parent
          比如現(xiàn)在Ext要加載了,它往上請(qǐng)求。如果最大的Bootstrap找不到,那么Boot會(huì)叫Ext自己找找,Ext找不到,是不會(huì)讓下一級(jí)的App去找的,此時(shí)就報(bào)出ClassNotFoundException

          4.類(lèi)A調(diào)用類(lèi)B,B會(huì)要求調(diào)用它的類(lèi)的類(lèi)加載器來(lái)加載它,也就是B會(huì)要求加載A的加載器來(lái)加載B。這就會(huì)有個(gè)問(wèn)題,如果他們?cè)谝黄穑菦](méi)關(guān)系,肯定某個(gè)classloader會(huì)把它們倆都加載好。但是如果A在/lib/ext文件夾中,而B(niǎo)在Classpath中呢?過(guò)程是這樣的首先加載A,那么一層層上到Bootstrap Classloader,boot沒(méi)找到所以ext自己找,找到了,沒(méi)問(wèn)題;加載B,因?yàn)锳調(diào)用了B,所以也從bootstrap來(lái)找,沒(méi)找到,然后A的ext classloader來(lái)找還是沒(méi)找到,但是再也不會(huì)往下調(diào)用了,于是報(bào)出ClassNotFoundException。
          但是現(xiàn)實(shí)生活中有很多應(yīng)用,比如JDBC核心方法在核心庫(kù)而驅(qū)動(dòng)在擴(kuò)展庫(kù),是必定在兩個(gè)地方的,那怎么辦呢?要用到Context ClassLoader我們?cè)诮⒁粋€(gè)線程Thread的時(shí)候,可以為這個(gè)線程通過(guò)setContextClassLoader方法來(lái)指定一個(gè)合適的classloader作為這個(gè)線程的context classloader,當(dāng)此線程運(yùn)行的時(shí)候,我們可以通過(guò)getContextClassLoader方法來(lái)獲得此context classloader,就可以用它來(lái)載入我們所需要的Class。默認(rèn)的是system classloader。利用這個(gè)特性,我們可以“打破”classloader委托機(jī)制了,父classloader可以獲得當(dāng)前線程的context classloader,而這個(gè)context classloader可以是它的子classloader或者其他的classloader,那么父classloader就可以從其獲得所需的 Class,這就打破了只能向父classloader請(qǐng)求的限制了。這個(gè)機(jī)制可以滿足當(dāng)我們的classpath是在運(yùn)行時(shí)才確定,并由定制的 classloader加載的時(shí)候,由system classloader(即在jvm classpath中)加載的class可以通過(guò)context classloader獲得定制的classloader并加載入特定的class(通常是抽象類(lèi)和接口,定制的classloader中是其實(shí)現(xiàn)),例如web應(yīng)用中的servlet就是用這種機(jī)制加載的.
          posted on 2008-02-20 15:16 離弦之ray的技術(shù)天空 閱讀(4845) 評(píng)論(4)  編輯  收藏 所屬分類(lèi): Java

          Feedback

          # re: Java ClassLoader基礎(chǔ)知識(shí)(網(wǎng)上內(nèi)容整理后所得) 2008-07-24 02:42 Yvon
          正在找這方面的資料,謝謝了  回復(fù)  更多評(píng)論
            

          # re: Java ClassLoader基礎(chǔ)知識(shí)(網(wǎng)上內(nèi)容整理后所得) 2009-05-19 13:31 tianma
          非常感謝,受益匪淺  回復(fù)  更多評(píng)論
            

          # re: Java ClassLoader基礎(chǔ)知識(shí)(網(wǎng)上內(nèi)容整理后所得) 2011-07-13 10:52 游客
          非常感謝。舉例說(shuō)明很清楚。  回復(fù)  更多評(píng)論
            

          # re: Java ClassLoader基礎(chǔ)知識(shí)(網(wǎng)上內(nèi)容整理后所得) 2011-10-20 11:18 @joe
          對(duì)classloader的理解又有了一點(diǎn)補(bǔ)充,謝謝  回復(fù)  更多評(píng)論
            

          主站蜘蛛池模板: 灯塔市| 吴忠市| 江源县| 库尔勒市| 井陉县| 高雄县| 工布江达县| 紫金县| 文登市| 兴业县| 东乡族自治县| 铜梁县| 福海县| 榆中县| 偃师市| 寿光市| 惠东县| 荆州市| 腾冲县| 赞皇县| 社会| 正定县| 板桥市| 乐至县| 简阳市| 西华县| 阿拉善左旗| 鄂尔多斯市| 东乌| 绥江县| 齐齐哈尔市| 锡林浩特市| 松溪县| 天祝| 大余县| 绥江县| 东光县| 扶沟县| 迭部县| 沙湾县| 精河县|