寫了這么多年的java代碼,異常一只是模模糊糊的氮惯,至今對異常類沒有個(gè)清楚的了解,所以再次總結(jié)一下想暗。
什么是運(yùn)行時(shí)異常呢妇汗,什么又是一般的異常呢,有什么區(qū)別呢说莫,什么時(shí)候該捕獲什么時(shí)候該拋出呢杨箭?
首先 exception和error都是繼承了throwable類 ,在java中只有throwable類型的實(shí)例才可以被拋出(throw)或者捕獲(catch)
exception和error你可以看到j(luò)ava對于異常處理的分類储狭。exception是程序運(yùn)行中可以預(yù)料的意外情況互婿,可能并且應(yīng)該被捕獲捣郊,進(jìn)行相應(yīng)的處理。
error是指正常情況下慈参,不大可能出現(xiàn)的情況呛牲,絕大部分都會導(dǎo)致程序(比如jvm自身)處于非正常,不可回復(fù)的狀態(tài)驮配。既然是非正常的娘扩,所以不便于也不需要捕獲,常見的有outofmemoryerror之類都是error的子類壮锻。
exception 又分為 可檢查(checked)異常和不檢查(unchecked)異常琐旁,可檢查異常在源代碼里必須顯示的進(jìn)行捕獲處理,這是編譯檢查的一部分猜绣。
不檢查異常就是所謂的運(yùn)行時(shí)異常灰殴,類似NullPointerException,ArrayIndexOutOfBoundsException之類掰邢,通澄眨可以編碼避免的邏輯錯誤,具體根據(jù)需要來判斷是否需要捕獲尸变,并不會在編譯期強(qiáng)制要求义图。
我們處理異常的時(shí)候要做到一下幾點(diǎn):
- 1:盡量不要捕獲類似exception這樣的通用異常,而是應(yīng)該捕獲特定異常召烂,軟件工程是門協(xié)作的藝術(shù)碱工,所以我們有義務(wù)讓自己的代碼能夠直觀的體現(xiàn)出盡量多的信息,而不是泛泛的exception之類奏夫,恰恰隱藏了我們的目的怕篷。另外,我們要保證程序不會捕獲到我們不希望捕獲的異常酗昼,比如廊谓,你可能跟希望runtimeexception被擴(kuò)散出來,而不是被捕獲麻削,進(jìn)一步講蒸痹,除非深思熟慮了,否則不要捕獲throwable或者error呛哟,這樣很難保證我們正確處理outmemoryerror叠荠。
- 2:不要生吞(swallow)異常。生吞異成ㄔ穑可能會導(dǎo)致非常難以診斷的詭異情況榛鼎。
我們從性能角度審視一下異常處理機(jī)制,這里有兩個(gè)可能會相對昂貴的地方:
- try-catch 代碼會產(chǎn)生額外的開銷,或者換個(gè)角度說者娱,它往往會影響jvm對代碼進(jìn)行優(yōu)化所以建議僅捕獲必要的代碼抡笼,盡量不要一個(gè)大的try包住整段的代碼,于此同時(shí)黄鳍,利用異惩埔觯控制代碼流程也不是一個(gè)好主意,遠(yuǎn)比我們通常意義上的提交語句(if/else际起,swithc)要低效拾碌。
-
java 每實(shí)例話一個(gè)exceptioon都會對當(dāng)時(shí)的棧進(jìn)行快照,這是一個(gè)相對比較重的操作街望,如果發(fā)生的比較頻繁校翔,這個(gè)開銷就不能被忽略了。
image.png