Java類(lèi)加載器特點(diǎn)的具體介紹
Java類(lèi)加載器一個(gè)我們經(jīng)常使用的優(yōu)秀語(yǔ)言,但是在我們使用的過(guò)程中有不少的問(wèn)題需要我們解決。下面我們就詳細(xì)的來(lái)看看相關(guān)問(wèn)題的解決方法。想大家有所收獲。
Java類(lèi)加載器的特性:
每個(gè)ClassLoader都維護(hù)了一份自己的名稱(chēng)空間, 同一個(gè)名稱(chēng)空間里不能出現(xiàn)兩個(gè)同名的類(lèi)。
為了實(shí)現(xiàn)java安全沙箱模型頂層的類(lèi)加載器安全機(jī)制, java默認(rèn)采用了 " 雙親委派的加載鏈 " 結(jié)構(gòu)。
類(lèi)圖中, BootstrapClassLoader是一個(gè)單獨(dú)的java類(lèi), 其實(shí)在這里, 不應(yīng)該叫他是一個(gè)java類(lèi)。因?yàn)椋呀?jīng)完全不用java實(shí)現(xiàn)了。它是在jvm啟動(dòng)時(shí), 就被構(gòu)造起來(lái)的, 負(fù)責(zé)java平臺(tái)核心庫(kù)。
自定義類(lèi)加載器加載一個(gè)類(lèi)的步驟
ClassLoader 類(lèi)加載邏輯分析, 以下邏輯是除 BootstrapClassLoader 外的類(lèi)加載器加載流程:
- // 檢查類(lèi)是否已被裝載過(guò)
- Class c = findLoadedClass(name);
- if (c == null ) {
- // 指定類(lèi)未被裝載過(guò)
- try {
- if (parent != null ) {
- // 如果父類(lèi)加載器不為空, 則委派給父類(lèi)加載
- c = parent.loadClass(name, false );
- } else {
- // 如果父類(lèi)加載器為空, 則委派給啟動(dòng)類(lèi)加載加載
- c = findBootstrapClass0(name);
- }
- } catch (ClassNotFoundException e) {
- // 啟動(dòng)類(lèi)加載器或父類(lèi)加載器拋出異常后, 當(dāng)前類(lèi)加載器將其
- // 捕獲, 并通過(guò)findClass方法, 由自身加載
- c = findClass(name);
- }
- }
線(xiàn)程上下文類(lèi)加載器
java默認(rèn)的線(xiàn)程上下文類(lèi)加載器是 系統(tǒng)類(lèi)加載器(AppClassLoader)。
- //Now create the class loader to use to launch the application
- try {
- loader = AppClassLoader.getAppClassLoader(extcl);
- } catch(IOException e) {
- throw new InternalError(
- "Could not create application class loader" );
- }
- //Also set the context class loader for the primordial thread.
- Thread.currentThread().setContextClassLoader(loader);
Java代碼
- //Now create the class loader to use to launch the application
- try {
- loader = AppClassLoader.getAppClassLoader(extcl);
- } catch (IOException e) {
- throw new InternalError(
- "Could not create application class loader" );
- }
- //Also set the context class loader for the primordial thread.
- Thread.currentThread().setContextClassLoader(loader);
以上代碼摘自sun.misc.Launch的無(wú)參構(gòu)造函數(shù)Launch()。使用線(xiàn)程上下文類(lèi)加載器, 可以在執(zhí)行線(xiàn)程中, 拋棄雙親委派加載鏈模式, 使用線(xiàn)程上下文里的類(lèi)加載器加載類(lèi).典型的例子有, 通過(guò)Java類(lèi)加載器上下文來(lái)加載第三方庫(kù)jndi實(shí)現(xiàn), 而不依賴(lài)于雙親委派.大部分java app服務(wù)器(jboss, tomcat..)也是采用contextClassLoader來(lái)處理web服務(wù)。還有一些采用 hotswap 特性的框架, 也使用了線(xiàn)程上下文類(lèi)加載器, 比如 seasar (full stack framework in japenese).#t#
線(xiàn)程上下文從根本解決了一般應(yīng)用不能違背雙親委派模式的問(wèn)題.使java類(lèi)加載體系顯得更靈活.隨著多核時(shí)代的來(lái)臨, 相信多線(xiàn)程開(kāi)發(fā)將會(huì)越來(lái)越多地進(jìn)入程序員的實(shí)際編碼過(guò)程中. 因此,在編寫(xiě)基礎(chǔ)設(shè)施時(shí), 通過(guò)使用線(xiàn)程上下文來(lái)加載類(lèi), 應(yīng)該是一個(gè)很好的選擇。當(dāng)然, 好東西都有利弊. 使用線(xiàn)程上下文加載類(lèi), 也要注意, 保證多根需要通信的線(xiàn)程間的類(lèi)加載器應(yīng)該是同一個(gè),防止因?yàn)椴煌念?lèi)加載器, 導(dǎo)致類(lèi)型轉(zhuǎn)換異常(ClassCastException)。
為什么要使用這種雙親委托模式呢?
因?yàn)檫@樣可以避免重復(fù)加載,當(dāng)父親已經(jīng)加載了該類(lèi)的時(shí)候,就沒(méi)有必要子ClassLoader再加載一次。以上就是對(duì)Java類(lèi)加載器的詳細(xì)介紹。





















