Posted on 2011-10-20 11:27
瘋狂 閱讀(1037)
評論(0) 編輯 收藏 所屬分類:
java
Java中一共有四個類加載器,之所以叫類加載器,是程序要用到某個類的時候,要用類加載器載入內(nèi)存。
這四個類加載器分別為:
Bootstrap ClassLoader、
Extension ClassLoader、
AppClassLoader和
URLClassLoader,他們的作用其實(shí)從名字就可以大概推測出來了。其中AppClassLoader在很多地方被叫做
System ClassLoader
Bootstrap ClassLoader是在JVM開始運(yùn)行的時候加載java的核心類,是用C++編寫的,它用來加載核心類庫,在JVM源代碼中這樣寫道:
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是用來加載擴(kuò)展類,即
/lib/ext中的類。
AppClassLoader用來加載
Classpath的類,是和我們關(guān)系最密切的類。
URLClassLoader用來加載網(wǎng)絡(luò)上遠(yuǎn)程的類,暫且不討論。
它們之間的關(guān)系:
1.
Parent-Child,按順序從大到小。不是簡單的繼承關(guān)系。
2.ClassLoader有個
getParent的方法,但是Ext ClassLoader調(diào)用后得到的是
null,bootstrap是JVM自己的,用戶看不到。
3.classloader的
委托機(jī)制:當(dāng)?shù)燃壉容^低的ClassLoader要加載某個類的時候,它首先會請求Parent加載器來加載,Parent再請求它的Parent
比如現(xiàn)在Ext要加載了,它往上請求。如果最大的Bootstrap找不到,那么Boot會叫Ext自己找找,Ext找不到,是
不會讓下一級的App去找的,此時就報出ClassNotFoundException
4.類A調(diào)用類B,B會要求調(diào)用它的類的類加載器來加載它,也就是B會要求加載A的加載器來加載B。這就會有個問題,如果他們在一起,那沒關(guān)系,肯定某個classloader會把它們倆都加載好。但是如果A在/lib/ext文件夾中,而B在Classpath中呢?過程是這樣的首先加載A,那么一層層上到Bootstrap Classloader,boot沒找到所以ext自己找,找到了,沒問題;加載B,因?yàn)锳調(diào)用了B,所以也從bootstrap來找,沒找到,然后A的ext classloader來找還是沒找到,但是再也不會往下調(diào)用了,于是報出ClassNotFoundException。
但是現(xiàn)實(shí)生活中有很多應(yīng)用,比如JDBC核心方法在核心庫而驅(qū)動在擴(kuò)展庫,是必定在兩個地方的,那怎么辦呢?要用到Context ClassLoader我們在建立一個線程Thread的時候,可以為這個線程通過setContextClassLoader方法來指定一個合適的classloader作為這個線程的context classloader,當(dāng)此線程運(yùn)行的時候,我們可以通過getContextClassLoader方法來獲得此context classloader,就可以用它來載入我們所需要的Class。默認(rèn)的是system classloader。利用這個特性,我們可以“打破”classloader委托機(jī)制了,父classloader可以獲得當(dāng)前線程的context classloader,而這個context classloader可以是它的子classloader或者其他的classloader,那么父classloader就可以從其獲得所需的 Class,這就打破了只能向父classloader請求的限制了。這個機(jī)制可以滿足當(dāng)我們的classpath是在運(yùn)行時才確定,并由定制的 classloader加載的時候,由system classloader(即在jvm classpath中)加載的class可以通過context classloader獲得定制的classloader并加載入特定的class(通常是抽象類和接口,定制的classloader中是其實(shí)現(xiàn)),例如web應(yīng)用中的servlet就是用這種機(jī)制加載的.
轉(zhuǎn)載自:
http://m.tkk7.com/clraychen/archive/2008/02/20/180868.html