<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    weidagang2046的專欄

    物格而后知致
    隨筆 - 8, 文章 - 409, 評(píng)論 - 101, 引用 - 0
    數(shù)據(jù)加載中……

    Java ClassLoader

     

    這是一篇較早時(shí)候?qū)懙奈恼?,最近?/FONT>J道看到一個(gè)與classloader有關(guān)的討論,于是重新翻出來(lái)。

    靜態(tài)庫(kù)、動(dòng)態(tài)連接庫(kù)

    程序編制一般需經(jīng)編輯、編譯、連接、加載和運(yùn)行幾個(gè)步驟。在我們的應(yīng)用中,有一些公共代碼是需要反復(fù)使用,就把這些代碼編譯為“庫(kù)”文件;在連接步驟中,連接器將從庫(kù)文件取得所需的代碼,復(fù)制到生成的可執(zhí)行文件中。這種庫(kù)稱為靜態(tài)庫(kù),其特點(diǎn)是可執(zhí)行文件中包含了庫(kù)代碼的一份完整拷貝;缺點(diǎn)就是被多次使用就會(huì)有多份冗余拷貝。

    為了克服這個(gè)缺點(diǎn)可以采用動(dòng)態(tài)連接庫(kù)。這個(gè)時(shí)候連接器僅僅是在可執(zhí)行文件中打上標(biāo)志,說(shuō)明需要使用哪些動(dòng)態(tài)連接庫(kù);當(dāng)運(yùn)行程序時(shí),加載器根據(jù)這些標(biāo)志把所需的動(dòng)態(tài)連接庫(kù)加載到內(nèi)存。

    另外在當(dāng)前的編程環(huán)境中,一般都提供方法讓程序在運(yùn)行的時(shí)候把某個(gè)特定的動(dòng)態(tài)連接庫(kù)加載并運(yùn)行,也可以將其卸載(例如Win32的LoadLibrary()&FreeLibrary()和Posix的dlopen()&dlclose())。這個(gè)功能被廣泛地用于在程序運(yùn)行時(shí)刻更新某些功能模塊或者是程序外觀。

    What is ClassLoader?

    與普通程序不同的是,Java程序(class文件)并不是本地的可執(zhí)行程序。當(dāng)運(yùn)行Java程序時(shí),首先運(yùn)行JVM(Java虛擬機(jī)),然后再把Java class加載到JVM里頭運(yùn)行,負(fù)責(zé)加載Java class的這部分就叫做Class Loader。

    JVM本身包含了一個(gè)ClassLoader稱為Bootstrap ClassLoader,和JVM一樣,Bootstrap ClassLoader是用本地代碼實(shí)現(xiàn)的,它負(fù)責(zé)加載核心Java Class(即所有java.*開頭的類)。另外JVM還會(huì)提供兩個(gè)ClassLoader,它們都是用Java語(yǔ)言編寫的,由Bootstrap ClassLoader加載;其中Extension ClassLoader負(fù)責(zé)加載擴(kuò)展的Java class(例如所有javax.*開頭的類和存放在JRE的ext目錄下的類),Application ClassLoader負(fù)責(zé)加載應(yīng)用程序自身的類。

    When to load the class?

    什么時(shí)候JVM會(huì)使用ClassLoader加載一個(gè)類呢?當(dāng)你使用java去執(zhí)行一個(gè)類,JVM使用Application ClassLoader加載這個(gè)類;然后如果類A引用了類B,不管是直接引用還是用Class.forName()引用,JVM就會(huì)找到加載類A的ClassLoader,并用這個(gè)ClassLoader來(lái)加載類B。

    Why use your own ClassLoader?

    似乎JVM自身的ClassLoader已經(jīng)足夠了,為什么我們還需要?jiǎng)?chuàng)建自己的ClassLoader呢?

    因?yàn)镴VM自帶的ClassLoader只是懂得從本地文件系統(tǒng)加載標(biāo)準(zhǔn)的java class文件,如果編寫你自己的ClassLoader,你可以做到:
    1)在執(zhí)行非置信代碼之前,自動(dòng)驗(yàn)證數(shù)字簽名
    2)動(dòng)態(tài)地創(chuàng)建符合用戶特定需要的定制化構(gòu)建類
    3)從特定的場(chǎng)所取得java class,例如數(shù)據(jù)庫(kù)中
    4) 等等

    事實(shí)上當(dāng)使用Applet的時(shí)候,就用到了特定的ClassLoader,因?yàn)檫@時(shí)需要從網(wǎng)絡(luò)上加載java class,并且要檢查相關(guān)的安全信息。

    目前的應(yīng)用服務(wù)器大都使用了ClassLoader技術(shù),即使你不需要?jiǎng)?chuàng)建自己的ClassLoader,了解其原理也有助于更好地部署自己的應(yīng)用。 

    ClassLoader Tree & Delegation Model

    當(dāng)你決定創(chuàng)建你自己的ClassLoader時(shí),需要繼承java.lang.ClassLoader或者它的子類。在實(shí)例化每個(gè)ClassLoader對(duì)象時(shí),需要指定一個(gè)父對(duì)象;如果沒(méi)有指定的話,系統(tǒng)自動(dòng)指定ClassLoader.getSystemClassLoader()為父對(duì)象。如下圖:

    在Java 1.2后,java class的加載采用所謂的委托模式(Delegation Modle),當(dāng)調(diào)用一個(gè)ClassLoader.loadClass()加載一個(gè)類的時(shí)候,將遵循以下的步驟:
    1)檢查這個(gè)類是否已經(jīng)被加載進(jìn)來(lái)了?
    2)如果還沒(méi)有加載,調(diào)用父對(duì)象加載該類
    3)如果父對(duì)象無(wú)法加載,調(diào)用本對(duì)象的findClass()取得這個(gè)類。

    所以當(dāng)創(chuàng)建自己的Class Loader時(shí),只需要重載findClass()這個(gè)方法。

    Unloading? Reloading?

    當(dāng)一個(gè)java class被加載到JVM之后,它有沒(méi)有可能被卸載呢?我們知道Win32有FreeLibrary()函數(shù),Posix有dlclose()函數(shù)可以被調(diào)用來(lái)卸載指定的動(dòng)態(tài)連接庫(kù),但是Java并沒(méi)有提供一個(gè)UnloadClass()的方法來(lái)卸載指定的類。

    在Java中,java class的卸載僅僅是一種對(duì)系統(tǒng)的優(yōu)化,有助于減少應(yīng)用對(duì)內(nèi)存的占用。既然是一種優(yōu)化方法,那么就完全是JVM自行決定如何實(shí)現(xiàn),對(duì)Java開發(fā)人員來(lái)說(shuō)是完全透明的。

    在什么時(shí)候一個(gè)java class/interface會(huì)被卸載呢?Sun公司的原話是這么說(shuō)的:"class or interface may be unloaded if and only if its class loader is unreachable. Classes loaded by the bootstrap loader may not be unloaded."

    事實(shí)上我們關(guān)心的不是如何卸載類的,我們關(guān)心的是如何更新已經(jīng)被加載了的類從而更新應(yīng)用的功能。JSP則是一個(gè)非常典型的例子,如果一個(gè)JSP文件被更改了,應(yīng)用服務(wù)器則需要把更改后的JSP重新編譯,然后加載新生成的類來(lái)響應(yīng)后繼的請(qǐng)求。

    其實(shí)一個(gè)已經(jīng)加載的類是無(wú)法被更新的,如果你試圖用同一個(gè)ClassLoader再次加載同一個(gè)類,就會(huì)得到異常(java.lang.LinkageError: duplicate class definition),我們只能夠重新創(chuàng)建一個(gè)新的ClassLoader實(shí)例來(lái)再次加載新類。至于原來(lái)已經(jīng)加載的類,開發(fā)人員不必去管它,因?yàn)樗赡苓€有實(shí)例正在被使用,只要相關(guān)的實(shí)例都被內(nèi)存回收了,那么JVM就會(huì)在適當(dāng)?shù)臅r(shí)候把不會(huì)再使用的類卸載。

    轉(zhuǎn)自:http://dev.csdn.net/article/68/68103.shtm

    posted on 2005-05-11 22:54 weidagang2046 閱讀(380) 評(píng)論(0)  編輯  收藏 所屬分類: Java

    主站蜘蛛池模板: 最好免费观看韩国+日本| 美国免费高清一级毛片| 西西人体免费视频| 亚洲自偷自偷在线制服| 国产福利免费视频| 亚洲区小说区激情区图片区| 七次郎成人免费线路视频| 亚洲免费日韩无码系列| 瑟瑟网站免费网站入口 | 18禁美女黄网站色大片免费观看| 国产AV无码专区亚洲精品| 你懂的免费在线观看| 亚洲小视频在线观看| 在线观看无码AV网站永久免费| 国产人成亚洲第一网站在线播放| 在线观看成人免费视频| 日本系列1页亚洲系列| 亚洲自偷自偷图片| 日本在线看片免费人成视频1000| 亚洲精品不卡视频| 国内一级一级毛片a免费| 无码的免费不卡毛片视频| 精品亚洲综合在线第一区| xxxxx免费视频| 久久亚洲中文字幕无码| 亚洲日韩国产精品第一页一区| 午夜免费福利片观看| 亚洲一区二区三区丝袜| jjzz亚洲亚洲女人| 久久大香伊焦在人线免费| 精品国产成人亚洲午夜福利| 免费吃奶摸下激烈视频| 日日麻批免费40分钟无码| 亚洲精品久久久久无码AV片软件| 亚洲а∨天堂久久精品| 美丽姑娘免费观看在线观看中文版 | 日本高清在线免费| 国产综合激情在线亚洲第一页| 久久久久久久尹人综合网亚洲| 一个人免费高清在线观看| 一级毛片不卡免费看老司机 |