首頁 考試吧論壇 Exam8視線 考試商城 網絡課程 模擬考試 考友錄 實用文檔 求職招聘 論文下載 | ||
![]() |
2011中考 | 2011高考 | 2012考研 | 考研培訓 | 在職研 | 自學考試 | 成人高考 | 法律碩士 | MBA考試 MPA考試 | 中科院 |
|
![]() |
四六級 | 職稱英語 | 商務英語 | 公共英語 | 托福 | 雅思 | 專四專八 | 口譯筆譯 | 博思 | GRE GMAT 新概念英語 | 成人英語三級 | 申碩英語 | 攻碩英語 | 職稱日語 | 日語學習 | 法語 | 德語 | 韓語 |
|
![]() |
計算機等級考試 | 軟件水平考試 | 職稱計算機 | 微軟認證 | 思科認證 | Oracle認證 | Linux認證 華為認證 | Java認證 |
|
![]() |
公務員 | 報關員 | 銀行從業資格 | 證券從業資格 | 期貨從業資格 | 司法考試 | 法律顧問 | 導游資格 報檢員 | 教師資格 | 社會工作者 | 外銷員 | 國際商務師 | 跟單員 | 單證員 | 物流師 | 價格鑒證師 人力資源 | 管理咨詢師考試 | 秘書資格 | 心理咨詢師考試 | 出版專業資格 | 廣告師職業水平 駕駛員 | 網絡編輯 |
|
![]() |
衛生資格 | 執業醫師 | 執業藥師 | 執業護士 | |
![]() |
會計從業資格考試(會計證) | 經濟師 | 會計職稱 | 注冊會計師 | 審計師 | 注冊稅務師 注冊資產評估師 | 高級會計師 | ACCA | 統計師 | 精算師 | 理財規劃師 | 國際內審師 |
|
![]() |
一級建造師 | 二級建造師 | 造價工程師 | 造價員 | 咨詢工程師 | 監理工程師 | 安全工程師 質量工程師 | 物業管理師 | 招標師 | 結構工程師 | 建筑師 | 房地產估價師 | 土地估價師 | 巖土師 設備監理師 | 房地產經紀人 | 投資項目管理師 | 土地登記代理人 | 環境影響評價師 | 環保工程師 城市規劃師 | 公路監理師 | 公路造價師 | 安全評價師 | 電氣工程師 | 注冊測繪師 | 注冊計量師 |
|
![]() |
繽紛校園 | 實用文檔 | 英語學習 | 作文大全 | 求職招聘 | 論文下載 | 訪談 | 游戲 |
1.1.1.4 類加載器(ClassLoader)
1.1.1.4.1 基礎知識
靜態庫、動態連接庫
程序編制一般需經編輯、編譯、連接、加載和運行幾個步驟。在我們的應用中,有一些公共代碼是需要反復使用,就把這些代碼編譯為“庫”文件;在連接步驟中,連接器將從庫文件取得所需的代碼,復制到生成的可執行文件中。這種庫稱為靜態庫,其特點是可執行文件中包含了庫代碼的一份完整拷貝;缺點就是被多次使用就會有多份冗余拷貝。
為了克服這個缺點可以采用動態連接庫。這個時候連接器僅僅是在可執行文件中打上標志,說明需要使用哪些動態連接庫;當運行程序時,加載器根據這些標志把所需的動態連接庫加載到內存。
另外在當前的編程環境中,一般都提供方法讓程序在運行的時候把某個特定的動態連接庫加載并運行,也可以將其卸載(例如Win32的LoadLibrary()&FreeLibrary()和Posix的dlopen()&dlclose())。這個功能被廣泛地用于在程序運行時刻更新某些功能模塊或者是程序外觀。
What is ClassLoader?
與普通程序不同的是,Java程序(class文件)并不是本地的可執行程序。當運行Java程序時,首先運行JVM(Java虛擬機),然后再把Java class加載到JVM里頭運行,負責加載Java class的這部分就叫做Class Loader。
JVM本身包含了一個ClassLoader稱為Bootstrap ClassLoader,和JVM一樣,Bootstrap ClassLoader是用本地代碼實現的,它負責加載核心Java Class(即所有java.*開頭的類)。另外JVM還會提供兩個ClassLoader,它們都是用Java語言編寫的,由Bootstrap ClassLoader加載;其中Extension ClassLoader負責加載擴展的Java class(例如所有javax.*開頭的類和存放在JRE的ext目錄下的類),Application ClassLoader負責加載應用程序自身的類。
When to load the class?
什么時候JVM會使用ClassLoader加載一個類呢?當你使用java去執行一個類,JVM使用Application ClassLoader加載這個類;然后如果類A引用了類B,不管是直接引用還是用Class.forName()引用,JVM就會找到加載類A的ClassLoader,并用這個ClassLoader來加載類B。
Why use your own ClassLoader?
似乎JVM自身的ClassLoader已經足夠了,為什么我們還需要創建自己的ClassLoader呢?
因為JVM自帶的ClassLoader只是懂得從本地文件系統加載標準的java class文件,如果編寫你自己的ClassLoader,你可以做到:
1)在執行非置信代碼之前,自動驗證數字簽名
2)動態地創建符合用戶特定需要的定制化構建類
3)從特定的場所取得java class,例如數據庫中
4) 等等
事實上當使用Applet的時候,就用到了特定的ClassLoader,因為這時需要從網絡上加載java class,并且要檢查相關的安全信息。
目前的應用服務器大都使用了ClassLoader技術,即使你不需要創建自己的ClassLoader,了解其原理也有助于更好地部署自己的應用。
ClassLoader Tree & Delegation Model
當你決定創建你自己的ClassLoader時,需要繼承java.lang.ClassLoader或者它的子類。在實例化每個ClassLoader對象時,需要指定一個父對象;如果沒有指定的話,系統自動指定ClassLoader.getSystemClassLoader()為父對象。如下圖:
在Java 1.2后,java class的加載采用所謂的委托模式(Delegation Modle),當調用一個ClassLoader.loadClass()加載一個類的時候,將遵循以下的步驟:
1)檢查這個類是否已經被加載進來了?
2)如果還沒有加載,調用父對象加載該類
3)如果父對象無法加載,調用本對象的findClass()取得這個類。
所以當創建自己的Class Loader時,只需要重載findClass()這個方法。
Unloading? Reloading?
當一個java class被加載到JVM之后,它有沒有可能被卸載呢?我們知道Win32有FreeLibrary()函數,Posix有dlclose()函數可以被調用來卸載指定的動態連接庫,但是Java并沒有提供一個UnloadClass()的方法來卸載指定的類。
在Java中,java class的卸載僅僅是一種對系統的優化,有助于減少應用對內存的占用。既然是一種優化方法,那么就完全是JVM自行決定如何實現,對Java開發人員來說是完全透明的。
在什么時候一個java class/interface會被卸載呢?Sun公司的原話是這么說的:"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."
事實上我們關心的不是如何卸載類的,我們關心的是如何更新已經被加載了的類從而更新應用的功能。JSP則是一個非常典型的例子,如果一個JSP文件被更改了,應用服務器則需要把更改后的JSP重新編譯,然后加載新生成的類來響應后繼的請求。
其實一個已經加載的類是無法被更新的,如果你試圖用同一個ClassLoader再次加載同一個類,就會得到異常(java.lang.LinkageError: duplicate class definition),我們只能夠重新創建一個新的ClassLoader實例來再次加載新類。至于原來已經加載的類,開發人員不必去管它,因為它可能還有實例正在被使用,只要相關的實例都被內存回收了,那么JVM就會在適當的時候把不會再使用的類卸載。
1.1.1.1.2 類加載的表現形式
java中的類是動態加載的,我們先看一下我們常用的類加載方式,先有一個感性的認識,才能進一步
深入討論,類加載無非就是下面三種方式。
class A{}
class B{}
class C{}
public class Loader{
public static void main(String[] args) throws Exception{
Class aa=A.class;
Class bb=Class.forName("B");
Class cc=ClassLoader.getSystemClassLoader().loadClass("C");
}
}
我們先看.class字面量方式,很多人可能不知道這種方式,因為這種用法不是一般java語法。
通過javap我們可以發現,這種方式的大致等價于定義了一個靜態成員變量
static Class class$0;(后面的編號是增長的)
你可以試圖再定義一個 static Class class$0,應該會收到一個編譯錯誤(重復定義)。
Class aa=A.class;
就相當于
if(class$0==null){
try{
Class.forName("A");
}
cacth(ClassNotFoundException e){
throw new NoClassDefFoundError(e);
}
}
Class aa=class$0;
可以很清楚的看到,這種類的字面量定義其實不是加載類的方式,而是被編譯器處理了,實質
上是使用了Class.forName方法,但是使用這種方式有一個很大的好處就是不用處理異常,因為
編譯器處理的時候如果找不到類會拋出一個NoClassDefFoundError。也許你覺得需要處理
ClassNotFoundException這種異常,事實上99%的情況下我們可以把這種異常認為是一個錯誤。
所以大部分情況我們使用這種方式會更簡潔。
最常用的方式就是Class.forName方式了,這也是一個通用的上層調用。這個方法有兩個重載,
可能很多人都忽略了第二個方法。
public static Class forName(String name) throws ClassNotFoundException
public static Class forName(String name, boolean initialize,ClassLoader loader) throws ClassNotFoundException
第二個方法后面多了兩個參數,第二個參數表示是否初始化,第三個參數為指定的類加載器。
在上面的例子中:
Class bb=Class.forName("B");等價于
Class bb=Class.forName("B",true,Loader.class.getClassLoader());
這里要詳細說一下這個類的初始化這個參數,如果這個參數為false的話,
類中的static成員不會被初始化,static語句塊也不會被執行。
也就是類雖然被加載了,但是沒有被初始化,不過在第一次使用時仍然會初始化。
所以我們有時候會看到Class.forName("XXX").newInstance()這樣的語句,為什么這里要創建一個
不用的實例呢?不過是為了保證類被初始化(兼容以前的系統)。
其實第二個方法是比較難用的,需要指定類加載器,如果不指定而且又沒有安裝安全管理器的化,
是無法加載類的,只要看一下具體的實現就明白了。
最本質的方式當然是直接使用ClassLoader加載了,所有的類最終都是通過ClassLoader加載的,
Class cc=ClassLoader.getSystemClassLoader().loadClass("C");
這里通過使用系統類加載器來加載某個類,很直接的方式,但是很遺憾的是通過這種方式加載類,
類是沒有被初始化的(也就是初始化被延遲到真正使用的時候).不過我們也可以借鑒上面的經驗,加載
后實例化一個對象Class cc=ClassLoader.getSystemClassLoader().loadClass("C").newInstance()。
這里使用了系統類加載器,也是最常用的類加載器,從classpath中尋找要加載的類。
java中默認有三種類加載器:引導類加載器,擴展類加載器,系統類加載器。
java中的類加載有著規范的層次結構,如果我們要了解類加載的過程,需要明確知道哪個類被誰
加載,某個類加載器加載了哪些類等等,就需要深入理解ClassLoader的本質。
以上只是類加載的表面的東西,我們還將討論深層次的東西。
相關推薦:2010年9月計算機等級考試試題及答案解析專題北京 | 天津 | 上海 | 江蘇 | 山東 |
安徽 | 浙江 | 江西 | 福建 | 深圳 |
廣東 | 河北 | 湖南 | 廣西 | 河南 |
海南 | 湖北 | 四川 | 重慶 | 云南 |
貴州 | 西藏 | 新疆 | 陜西 | 山西 |
寧夏 | 甘肅 | 青海 | 遼寧 | 吉林 |
黑龍江 | 內蒙古 |