Java基礎(chǔ)知識(shí)系列(二)ClassNotFoundException 和 NoClassDefFoundError的區(qū)別

原文傳送門(mén)

這同時(shí)也是一個(gè)經(jīng)常被問(wèn)到的面試題徒役。

字面上的區(qū)別

一個(gè)是Error粒褒,一個(gè)是Exception哗蜈。在Java中Error和Exception是有區(qū)別的:我們可以從Exception中恢復(fù)程序茁瘦,但卻不應(yīng)該嘗試從Error中恢復(fù)程序纽竣。

產(chǎn)生的原因

ClassNotFoundException產(chǎn)生的原因

源碼注釋

JDK8的源碼注釋是這樣寫(xiě)的:

Thrown when an application tries to load in a class through its string name using:
- The forName method in class Class
- The findSystemClass method in class ClassLoader .
- The loadClass method in class ClassLoader.
but no definition for the class with the specified name could be found.

翻譯一下

當(dāng)應(yīng)用程序嘗試使用以下命令通過(guò)其字符串名稱(chēng)加載類(lèi)時(shí):
 - 類(lèi)Class中的forName方法墓贿。
 - 類(lèi)ClassLoader中的findSystemClass方法茧泪。
 - 類(lèi)ClassLoader中的loadClass方法。
這時(shí)找不到具有指定名稱(chēng)的類(lèi)的定義聋袋。這時(shí)拋出ClassNotFoundException異常队伟。

以Class.forName方法為例,任意一個(gè)類(lèi)的類(lèi)名如果被作為參數(shù)傳幽勒,遞給這個(gè)方法都將導(dǎo)致該類(lèi)被加載到 JVM 中嗜侮。如果這個(gè)類(lèi)在類(lèi)路徑中沒(méi)有被找到,那么此時(shí)就會(huì)在運(yùn)行時(shí)拋出 ClassNotFoundException 異常啥容。

解決方法

要解決這個(gè)問(wèn)題锈颗,就要確保所需的類(lèi)連同它依賴(lài)的包存在于類(lèi)路徑中。當(dāng) Class.forName 被調(diào)用的時(shí)候咪惠,類(lèi)加載器會(huì)查找類(lèi)路徑中的類(lèi)击吱,如果找到了那么這個(gè)類(lèi)就會(huì)被成功加載,如果沒(méi)找到遥昧,那么就會(huì)拋出ClassNotFountException覆醇。

根據(jù)注釋?zhuān)瑫?huì)拋出這個(gè)異常的有三個(gè)方法:
Class.forName、ClassLoader.loadClass和ClassLOader.findSystemClass炭臭。

所以永脓,類(lèi)的動(dòng)態(tài)加載有可能引發(fā)ClassNotFoundException異常時(shí),我們可以在此處catch這個(gè)異常來(lái)處理鞋仍。

NoClassDefFoundError產(chǎn)生的原因

源碼注釋

同樣憨奸,來(lái)看一下源碼注釋

Thrown if the Java Virtual Machine or a *ClassLoader* instance tries to load in the definition of a class (as part of a normal method call or as part of creating a new instance using the *new* expression) and no definition of the class could be found.
The searched-for class definition existed when the currently executing class was compiled, but the definition can no longer be found.

翻譯:

當(dāng) Java 虛擬機(jī) 或 ClassLoader 實(shí)例試圖在類(lèi)的定義中加載(作為通常方法調(diào)用的一部分,或者是使用 new 來(lái)創(chuàng)建新的對(duì)象)時(shí)凿试,卻找不到類(lèi)的定義(要查找的類(lèi)在編譯的時(shí)候是存在的排宰,運(yùn)行的時(shí)候卻找不到了),拋出此異常那婉。
即當(dāng)前執(zhí)行的類(lèi)被編譯時(shí)板甘,所搜索的類(lèi)定義存在,但無(wú)法再找到該定義详炬。 

這是因?yàn)镴ava虛擬機(jī)在編譯時(shí)能找到合適的類(lèi)盐类,而在運(yùn)行時(shí)不能找到合適的類(lèi)導(dǎo)致的錯(cuò)誤。

由于 NoClassDefFoundError 是有 JVM 引起的呛谜,所以不應(yīng)該嘗試捕捉這個(gè)錯(cuò)誤在跳。

解決方法

查找那些在開(kāi)發(fā)期間存在于類(lèi)路徑下但在運(yùn)行期間卻不在類(lèi)路徑下的類(lèi)。
NoClassDefFoundError的錯(cuò)誤是由于在運(yùn)行時(shí)類(lèi)加載器在classpath下找不到需要加載的類(lèi)隐岛,所以我們需要把對(duì)應(yīng)的類(lèi)加載到classpath中猫妙,或者檢查為什么類(lèi)在classpath中是不可用的。

兩者的區(qū)別到底是什么聚凹?

ClassNotFoundException 發(fā)生在類(lèi)動(dòng)態(tài)裝入階段割坠。

當(dāng)應(yīng)用程序試圖通過(guò)類(lèi)的字符串名稱(chēng)齐帚,使用常規(guī)的三種方法裝入類(lèi),但卻找不到指定名稱(chēng)的類(lèi)定義時(shí)就拋出該異常彼哼。

NoClassDefFoundError發(fā)生在運(yùn)行階段对妄。
在運(yùn)行時(shí)我們想調(diào)用某個(gè)類(lèi)的方法或者訪(fǎng)問(wèn)這個(gè)類(lèi)的靜態(tài)成員的時(shí)候,發(fā)現(xiàn)這個(gè)類(lèi)不可用敢朱,此時(shí)Java虛擬機(jī)就會(huì)拋出NoClassDefFoundError錯(cuò)誤剪菱。

總結(jié)一下,簡(jiǎn)單來(lái)說(shuō)拴签,NoClassDefFoundError和ClassNotFoundException都是由于在CLASSPATH下找不到對(duì)應(yīng)的類(lèi)而引起的琅豆,通常是缺少對(duì)應(yīng)的jar包,不過(guò)篓吁,JVM認(rèn)為:
(1)當(dāng)應(yīng)用運(yùn)行時(shí)沒(méi)有找到對(duì)應(yīng)的引用茫因,則會(huì)拋出java.lang.NoClassDefFoundError;
(2)當(dāng)你在代碼中顯式加載類(lèi)(使用Class.forName())時(shí)沒(méi)有找到對(duì)應(yīng)的類(lèi)杖剪,則會(huì)拋出java.lang.ClassNotFoundException冻押。
開(kāi)發(fā)者經(jīng)常遇到的情況是:ClassNotFoundException異常引起了ClassNoDefFoundError。

附上兩張很有用的圖


jvm加載類(lèi)的階段
ClassNotFoundException和NoClassDefFoundError的區(qū)別
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末盛嘿,一起剝皮案震驚了整個(gè)濱河市洛巢,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌次兆,老刑警劉巖稿茉,帶你破解...
    沈念sama閱讀 211,194評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異芥炭,居然都是意外死亡漓库,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,058評(píng)論 2 385
  • 文/潘曉璐 我一進(jìn)店門(mén)园蝠,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)渺蒿,“玉大人,你說(shuō)我怎么就攤上這事彪薛∶埃” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 156,780評(píng)論 0 346
  • 文/不壞的土叔 我叫張陵善延,是天一觀(guān)的道長(zhǎng)少态。 經(jīng)常有香客問(wèn)我,道長(zhǎng)易遣,這世上最難降的妖魔是什么彼妻? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,388評(píng)論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮训挡,結(jié)果婚禮上澳骤,老公的妹妹穿的比我還像新娘。我一直安慰自己澜薄,他們只是感情好为肮,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,430評(píng)論 5 384
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著肤京,像睡著了一般颊艳。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上忘分,一...
    開(kāi)封第一講書(shū)人閱讀 49,764評(píng)論 1 290
  • 那天棋枕,我揣著相機(jī)與錄音,去河邊找鬼妒峦。 笑死重斑,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的肯骇。 我是一名探鬼主播窥浪,決...
    沈念sama閱讀 38,907評(píng)論 3 406
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼笛丙!你這毒婦竟也來(lái)了漾脂?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 37,679評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤胚鸯,失蹤者是張志新(化名)和其女友劉穎骨稿,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體姜钳,經(jīng)...
    沈念sama閱讀 44,122評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡坦冠,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,459評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了哥桥。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片蓝牲。...
    茶點(diǎn)故事閱讀 38,605評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖泰讽,靈堂內(nèi)的尸體忽然破棺而出例衍,到底是詐尸還是另有隱情,我是刑警寧澤已卸,帶...
    沈念sama閱讀 34,270評(píng)論 4 329
  • 正文 年R本政府宣布佛玄,位于F島的核電站,受9級(jí)特大地震影響累澡,放射性物質(zhì)發(fā)生泄漏梦抢。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,867評(píng)論 3 312
  • 文/蒙蒙 一愧哟、第九天 我趴在偏房一處隱蔽的房頂上張望奥吩。 院中可真熱鬧哼蛆,春花似錦、人聲如沸霞赫。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,734評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)端衰。三九已至叠洗,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間旅东,已是汗流浹背灭抑。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,961評(píng)論 1 265
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留抵代,地道東北人腾节。 一個(gè)月前我還...
    沈念sama閱讀 46,297評(píng)論 2 360
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像荤牍,于是被迫代替她去往敵國(guó)和親禀倔。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,472評(píng)論 2 348