寫了這么多年的java代碼,異常一只是模模糊糊的,至今對(duì)異常類沒(méi)有個(gè)清楚的了解,所以再次總結(jié)一下。
什么是運(yùn)行時(shí)異常呢,什么又是一般的異常呢,有什么區(qū)別呢,什么時(shí)候該捕獲什么時(shí)候該拋出呢?
首先 exception和error都是繼承了throwable類 ,在java中只有throwable類型的實(shí)例才可以被拋出(throw)或者捕獲(catch)
exception和error你可以看到j(luò)ava對(duì)于異常處理的分類。exception是程序運(yùn)行中可以預(yù)料的意外情況,可能并且應(yīng)該被捕獲,進(jìn)行相應(yīng)的處理。
error是指正常情況下,不大可能出現(xiàn)的情況,絕大部分都會(huì)導(dǎo)致程序(比如jvm自身)處于非正常,不可回復(fù)的狀態(tài)。既然是非正常的,所以不便于也不需要捕獲,常見的有outofmemoryerror之類都是error的子類。
exception 又分為 可檢查(checked)異常和不檢查(unchecked)異常,可檢查異常在源代碼里必須顯示的進(jìn)行捕獲處理,這是編譯檢查的一部分。
不檢查異常就是所謂的運(yùn)行時(shí)異常,類似NullPointerException,ArrayIndexOutOfBoundsException之類,通??梢跃幋a避免的邏輯錯(cuò)誤,具體根據(jù)需要來(lái)判斷是否需要捕獲,并不會(huì)在編譯期強(qiáng)制要求。
我們處理異常的時(shí)候要做到一下幾點(diǎn):
- 1:盡量不要捕獲類似exception這樣的通用異常,而是應(yīng)該捕獲特定異常,軟件工程是門協(xié)作的藝術(shù),所以我們有義務(wù)讓自己的代碼能夠直觀的體現(xiàn)出盡量多的信息,而不是泛泛的exception之類,恰恰隱藏了我們的目的。另外,我們要保證程序不會(huì)捕獲到我們不希望捕獲的異常,比如,你可能跟希望runtimeexception被擴(kuò)散出來(lái),而不是被捕獲,進(jìn)一步講,除非深思熟慮了,否則不要捕獲throwable或者error,這樣很難保證我們正確處理outmemoryerror。
- 2:不要生吞(swallow)異常。生吞異??赡軙?huì)導(dǎo)致非常難以診斷的詭異情況。
我們從性能角度審視一下異常處理機(jī)制,這里有兩個(gè)可能會(huì)相對(duì)昂貴的地方:
- try-catch 代碼會(huì)產(chǎn)生額外的開銷,或者換個(gè)角度說(shuō),它往往會(huì)影響jvm對(duì)代碼進(jìn)行優(yōu)化所以建議僅捕獲必要的代碼,盡量不要一個(gè)大的try包住整段的代碼,于此同時(shí),利用異??刂拼a流程也不是一個(gè)好主意,遠(yuǎn)比我們通常意義上的提交語(yǔ)句(if/else,swithc)要低效。
-
java 每實(shí)例話一個(gè)exceptioon都會(huì)對(duì)當(dāng)時(shí)的棧進(jìn)行快照,這是一個(gè)相對(duì)比較重的操作,如果發(fā)生的比較頻繁,這個(gè)開銷就不能被忽略了。
image.png
