ExtClassLoader與AppClassLoader都是 java.net.URLClassLoader的子類別,可以在使用java啟動(dòng)程式時(shí),使用以下的指令來指定ExtClassLoader的搜尋路徑:
java -Djava.ext.dirs=c:\workspace\ YourClass
可以在使用java啟動(dòng)程式時(shí),使用-classpath或-cp來指定AppClassLoader的搜尋路徑,也就是設(shè)定Classpath:
java -classpath c:\workspace\ YourClass
ExtClassLoader與AppClassLoader在程式啟動(dòng)後會(huì)在虛擬機(jī)器中存在一份,在程式運(yùn)行過程中就無法再改變它的搜尋路徑,如果在程式運(yùn)行過程中,打算動(dòng)態(tài)決定從其它的路徑載入類別,就要產(chǎn)生新的類別載入器。
可以使用URLClassLoader來產(chǎn)生新的類別載入器,它需要java.net.URL作為其參數(shù)來指定類別載入的搜尋路徑,例如:
URL url = new URL("file:/d:/workspace/");
ClassLoader urlClassLoader = new URLClassLoader(new URL[] {url});
Class c = urlClassLoader.loadClass("SomeClass");
由 於ClassLoader是Java SE的標(biāo)準(zhǔn)API之一,可以在rt.jar中找到,因而會(huì)由Bootstrap Loader來載入ClassLoader類別,在新增了ClassLoader實(shí)例後,可以使用它的loadClass()方法來指定要載入的類別名 稱,在新增ClassLoader時(shí),會(huì)自動(dòng)將新建的ClassLoader的parent設(shè)定為AppClassLoader,並在每次載入類別時(shí),先 委託 parent代為搜尋,所以上例中搜尋SomeClass類別時(shí),會(huì)一路往上委託至Bootstrap Loader先開始搜尋,接著是ExtClassLoader、AppClassLoader,如果都找不到,才使用新建的ClassLoader搜尋。
Java 的類別載入器階層架構(gòu)除了可以達(dá)到動(dòng)態(tài)載入類別目的之外,還有著安全上的考量,首先,因?yàn)槊看螌ふ翌悇e時(shí)都是委託parent開始尋找,所以除非有人可以 侵入你的電腦,置換掉標(biāo)準(zhǔn)Java SE API與您自己安裝的延伸套件,否則是不可能藉由撰寫自己的類別載入器來載入惡意類別,以置換掉標(biāo)準(zhǔn)Java SE API與您自己安裝的延伸套件。
由於每次的類別載入是由子ClassLoader委託父ClassLoader先嘗試載入,但父ClassLoader看不到子ClassLoader,所以同一階層的子ClassLoader不會(huì)被誤用,從而避免了載入錯(cuò)誤類別的可能性。
由 同一個(gè)ClassLoader載入的類別檔案,會(huì)只有一份Class實(shí)例,如果同一個(gè)類別檔案是由兩個(gè)不同的ClassLoader載入,則會(huì)有兩份不同 的Class實(shí)例。注意這個(gè)說法,如果有兩個(gè)不同的ClassLoader搜尋同一個(gè)類別,而在parent的 AppClassLoader搜尋路徑中就可以找到指定類別的話,則Class實(shí)例就只會(huì)有一個(gè),因?yàn)閮蓚€(gè)不同的ClassLoader都是在委託父 ClassLoader時(shí)找到該類別的,如果父ClassLoader找不到,而是由各自的ClassLoader搜尋到,則Class的實(shí)例會(huì)有兩份。
以下範(fàn)例是個(gè)簡單示範(fàn),可用來測試載入路徑與Class實(shí)例是否為同一物件:
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLClassLoader;
public class ClassLoaderDemo {
public static void main(String[] args) {
try {
// 測試路徑
String classPath = args[0];
// 測試類別
String className = args[1];
URL url1 = new URL(classPath);
// 建立ClassLoader
ClassLoader loader1 = new URLClassLoader(new URL[] {url1});
// 載入指定類別
Class c1 = loader1.loadClass(className);
// 顯示類別描述
System.out.println(c1);
URL url2 = new URL(classPath);
ClassLoader loader2 = new URLClassLoader(new URL[] {url2});
Class c2 = loader2.loadClass(className);
System.out.println(c2);
System.out.println("c1 與 c1 為同一實(shí)例?" + (c1 == c2));
}
catch(ArrayIndexOutOfBoundsException e) {
System.out.println("沒有指定類別載入路徑與名稱");
}
catch(MalformedURLException e) {
System.out.println("載入路徑錯(cuò)誤");
}
catch(ClassNotFoundException e) {
System.out.println("找不到指定的類別");
}
}
}
你 可以任意設(shè)計(jì)一個(gè)類別,例如TestClass,其中classPath可以輸入不為ExtClassLoader或AppClassLoader的搜尋 路徑,例如file:/d:/workspace/,這樣同一個(gè)類別會(huì)分由兩個(gè)ClassLoader載入,結(jié)果會(huì)有兩份Class實(shí)例,則測試c1與 c2是否為同一實(shí)例時(shí),則結(jié)果會(huì)顯示false,一個(gè)執(zhí)行結(jié)果如下:
java ClassLoaderDemo file:/d:/workspace/ TestClass
class TestClass
class TestClass
c1 與 c1 為同一實(shí)例?false
如 果在執(zhí)行程式時(shí),以-cp將file:/d:/workspace/加入為Classpath的一部份,由於兩個(gè)ClassLoader的parent都 是AppClassLoader,而AppClassLoader會(huì)在Classpath中找到指定的類別,所以最後會(huì)只有一個(gè)指定的類別之Class實(shí) 例,則測試c1與c2是否為同一實(shí)例時(shí),結(jié)果會(huì)顯示true,一個(gè)執(zhí)行結(jié)果如下:
java -cp .;d:\workspace ClassLoaderDemo file:/d:/workspace/ TestClass
class TestClass
class TestClass
c1 與 c1 為同一實(shí)例?true
在 不同的環(huán)境中,應(yīng)用程式可能會(huì)設(shè)定自己的類別載入器,例如在Tomcat的類別載入器,會(huì)找尋Tomcat目錄中l(wèi)ib中的jar檔案之類別,而Web應(yīng) 用程式也會(huì)從WEB-INF的lib中找尋jar檔案,以及從WEB-INF/classes中找尋.class檔,搞清楚類別載入器載入檔案的位置與順 序,遇到ClassNotFoundException或是NoClassDefFoundError時(shí),才會(huì)知道要在哪邊確認(rèn)類別檔案是否存在。