??? 在java中,類(lèi)裝載異常雖然不是很常見(jiàn),可如果發(fā)生異常,其處理還是比較麻煩的。
??? 下面的幾種異常是比較常見(jiàn)的:
ClassNotFoundException
??? ClassNotFoundException 是最常見(jiàn)的類(lèi)裝入異常類(lèi)型。它發(fā)生在裝入階段。Java 規(guī)范對(duì) ClassNotFoundException 的描述是這樣的:
??? 當(dāng)應(yīng)用程序試圖通過(guò)類(lèi)的字符串名稱(chēng),使用以下三種方法裝入類(lèi),但卻找不到指定名稱(chēng)的類(lèi)定義時(shí)拋出該異常。
??? * 類(lèi) Class 中的 forName() 方法。
??? * 類(lèi) ClassLoader 中的 findSystemClass() 方法。
??? * 類(lèi) ClassLoader 中的 loadClass() 方法。
??? 所以,如果顯式地裝入類(lèi)的嘗試失敗,那么就拋出 ClassNotFoundException。
??? 這些異常修復(fù)起來(lái)通常比較簡(jiǎn)單??梢杂? verbose 選項(xiàng)檢查類(lèi)路徑,確保使用的類(lèi)路徑設(shè)置正確。如果類(lèi)路徑設(shè)置正確,但是仍然看到這個(gè)錯(cuò)誤,那么就是需要的類(lèi)在類(lèi)路徑中不存在。要修復(fù)這個(gè)問(wèn)題,可以把類(lèi)移動(dòng)到類(lèi)路徑中指定的目錄或 JAR 文件中,或者把類(lèi)所在的位置添加到類(lèi)路徑中。
NoClassDefFoundError
??? NoClassDefFoundError 是類(lèi)裝入器在裝入階段拋出的另一個(gè)常見(jiàn)異常。JVM 規(guī)范對(duì) NoClassDefFoundError 的定義如下:
??? 如果 Java 虛擬機(jī)或 ClassLoader 實(shí)例試圖裝入類(lèi)定義(作為正常的方法調(diào)用的一部分,或者作為使用 new 表達(dá)式創(chuàng)建新實(shí)例的一部分),但卻沒(méi)有找到類(lèi)定義時(shí)拋出該異常。
??? 當(dāng)目前執(zhí)行的類(lèi)已經(jīng)編譯,但是找不到它的定義時(shí),會(huì)存在 searched-for 類(lèi)定義。
??? 實(shí)際上,這意味著 NoClassDefFoundError 的拋出,是不成功的隱式類(lèi)裝入的結(jié)果。
??? 簡(jiǎn)單說(shuō)來(lái),就是引用的類(lèi)在類(lèi)路徑中沒(méi)有找到。
ClassCastException
??? 類(lèi)裝入器能夠拋出的另一個(gè)異常是 ClassCastException。它是在類(lèi)型比較中發(fā)現(xiàn)不兼容類(lèi)型的時(shí)候拋出的。JVM 規(guī)范指定 ClassCastException 是:
??? 該異常的拋出,表明代碼企圖把對(duì)象的類(lèi)型轉(zhuǎn)換成一個(gè)子類(lèi),而該對(duì)象并不是這個(gè)子類(lèi)的實(shí)例。
UnsatisfiedLinkError
??? 在把本地方法調(diào)用鏈接到對(duì)應(yīng)的本機(jī)定義時(shí),類(lèi)裝入器扮演著重要角色。如果程序試圖裝入一個(gè)不存在或者放錯(cuò)的本機(jī)庫(kù)時(shí),在鏈接階段的解析過(guò)程會(huì)發(fā)生 UnsatisfiedLinkError。JVM 規(guī)范指定 UnsatisfiedLinkError 是:
??? 對(duì)于聲明為 native 的方法,如果 Java 虛擬機(jī)找不到和它對(duì)應(yīng)的本機(jī)語(yǔ)言定義,就會(huì)拋出該異常。
??? 當(dāng)調(diào)用本機(jī)方法時(shí),類(lèi)裝入器會(huì)嘗試裝入定義了該方法的本機(jī)庫(kù)。如果找不到這個(gè)庫(kù),就會(huì)拋出這個(gè)錯(cuò)誤。
??? 本機(jī)庫(kù)的裝入由調(diào)用 System.loadLibrary() 方法的類(lèi)的類(lèi)裝入器啟動(dòng) ,根據(jù)使用的類(lèi)裝入器,會(huì)搜索不同的位置:
??? * 對(duì)于由 bootstrap 類(lèi)裝入器裝入的類(lèi),搜索 sun.boot.library.path。
??? * 對(duì)于由擴(kuò)展類(lèi)裝入器裝入的類(lèi),先搜索 java.ext.dirs,然后是 sun.boot.library.path,然后是 java.library.path。
??? * 對(duì)于由系統(tǒng)類(lèi)裝入器裝入的類(lèi),搜索 sun.boot.library.path,然后是 java.library.path。
ClassCircularityError
??? JVM 規(guī)范指定 ClassCircularityError 的拋出條件是:
??? 類(lèi)或接口由于是自己的超類(lèi)或超接口而不能被裝入。
??? 這個(gè)錯(cuò)誤是在鏈接階段的解析過(guò)程中拋出的。這個(gè)錯(cuò)誤有點(diǎn)奇怪,因?yàn)?Java 編譯器不允許發(fā)生這種循環(huán)情況。但是,如果獨(dú)立地編譯類(lèi),然后再把它們放在一起,就可能發(fā)生這個(gè)錯(cuò)誤。
ClassFormatError
JVM 規(guī)范指出,拋出 ClassFormatError 的條件是:
??? 負(fù)責(zé)指定所請(qǐng)求的編譯類(lèi)或接口的二進(jìn)制數(shù)據(jù)形式有誤。
??? 這個(gè)異常是在類(lèi)裝入的鏈接階段的校驗(yàn)過(guò)程中拋出。如果字節(jié)碼發(fā)生了更改,例如主版本號(hào)或次版本號(hào)發(fā)生了更改,那么二進(jìn)制數(shù)據(jù)的形式就會(huì)有誤。例如,如果對(duì)字節(jié)碼故意做了更改,或者在通過(guò)網(wǎng)絡(luò)傳送類(lèi)文件時(shí)現(xiàn)出了錯(cuò)誤,那么就可能發(fā)生這個(gè)異常。
??? 修復(fù)這個(gè)問(wèn)題的惟一方法就是獲得字節(jié)碼的正確副本,可能需要重新進(jìn)行編譯。
ExceptionInInitializerError
根據(jù) JVM 規(guī)范,拋出 ExceptionInInitializer 的情況是:
??? * 如果初始化器突然完成,拋出一些異常 E,而且 E 的類(lèi)不是 Error 或者它的某個(gè)子類(lèi),那么就會(huì)創(chuàng)建 ExceptionInInitializerError 類(lèi)的一個(gè)新實(shí)例,并用 E 作為參數(shù),用這個(gè)實(shí)例代替 E。
??? * 如果 Java 虛擬機(jī)試圖創(chuàng)建類(lèi) ExceptionInInitializerError 的新實(shí)例,但是因?yàn)槌霈F(xiàn) Out-Of-Memory-Error 而無(wú)法創(chuàng)建新實(shí)例,那么就拋出 OutOfMemoryError 對(duì)象作為代替。