很早時就關注了classloader,但一直沒有總結
1.classloader是樹形結構。
bootstrap Class Loaders負責裝載java.*下的基本類
extension Class Loaders負責裝載javax.*下的類
system Class Loaders負責系統(用戶)實現的類
三者的關系是
bootstrap class loaders是extension class loaders的父親
extension class loaders是system class loaders的父親
2.class loader的裝載機制是parent delegate的模型。類的裝載是委托給父class loader去查找,如果沒有找到才用當前的class loader來查找。
3.不同的classLoader加載同一個類,實例是不同的,拋出castclassException
4.Class.forName是從指定的classloader中裝載類,如果沒有指定,也就是一個參數的時候,是從裝載當前對象實例所在的classloader中裝載類. 而ClassLoader的實例調用loadclass方法,是指從當前ClassLoader實例中調用類,而這個實例與裝載當前所在類實例的Classloader也許不是同一個. 舉個例子吧, 有A,B , C兩個ClassLoader , 當前運行的類D的實例是d(裝載它的是A) , 如果D中使用Class.forName那么就是使用的ClassLoader就是A,當然,也可以指定為B. 而如果D中代碼找到的ClassLoader實例是C,那么就是用D來裝載所指定的類.
5.Thread.currentThread().getContextClassLoader().loadClass("className"),采用當前線程的類加載器
比如:
1
class A
{
2
..
3
public void test()
{
4
..
5
Class B =Thread.currentThread().getContextClassLoader().loadClass("className");
6
..
7
}
8
..
9
}
如果將第5行中的的語句換成
1
Class B=Class.forName("className")
這兩種情況下calss B是一致的么?
回答:大多數情況下,是一樣的,但是如果不改變 Thread的ClassLoader ,那么是否也是一樣的呢?
考慮Java多線程應用,執行類A的方法體B中采用Tread的方式獲得classloader是調用者的類加載器。而class.forname是加載當前類的也就是類A的類加載器。這兩種情況下是可能不一致的。
在spring的環境下,類C中的類A實例通過依賴注入,如果采用Class.forName則Class B是IoC容器的類加載器;如果采用thread的話,則類加載器和加載類C的一樣