java異常處理的經(jīng)驗(yàn)總結(jié)

這篇文章主要是對(duì)Java異常選擇和使用中的一些誤區(qū)的總結(jié)和歸納,希望各位讀者能夠熟練掌握異常處理的一些注意點(diǎn)和原則。只有處理好了異常,才能提升開發(fā)人員的基本素養(yǎng),提高系統(tǒng)的健壯性劫拢,提升用戶體驗(yàn),提高產(chǎn)品的價(jià)值胖缤。廢話少說尚镰,直接看:

誤區(qū)一、異常的選擇

這張圖描述了異常的結(jié)構(gòu)哪廓,其實(shí)我們都知道異常分檢測(cè)異常和非檢測(cè)異常狗唉,但是在實(shí)際中又混淆了這兩種異常的應(yīng)用。由于非檢測(cè)異常使用方便涡真,很多開發(fā)人員就認(rèn)為檢測(cè)異常沒什么用處分俯。其實(shí)異常的應(yīng)用情景可以概括為以下:

1.調(diào)用代碼不能繼續(xù)執(zhí)行,需要立即終止哆料。出現(xiàn)這種情況的可能性太多太多缸剪,例如服務(wù)器連接不上、參數(shù)不正確等东亦。這些時(shí)候都適用非檢測(cè)異常杏节,不需要調(diào)用代碼的顯式捕捉和處理,而且代碼簡潔明了典阵。

2.調(diào)用代碼需要進(jìn)一步處理和恢復(fù)奋渔。假如將 SQLException 定義為非檢測(cè)異常,這樣操作數(shù)據(jù)時(shí)開發(fā)人員理所當(dāng)然的認(rèn)為 SQLException 不需要調(diào)用代碼的顯式捕捉和處理壮啊,進(jìn)而會(huì)導(dǎo)致嚴(yán)重的 Connection 不關(guān)閉嫉鲸、Transaction 不回滾、DB 中出現(xiàn)臟數(shù)據(jù)等情況歹啼,正因?yàn)?SQLException 定義為檢測(cè)異常玄渗,才會(huì)驅(qū)使開發(fā)人員去顯式捕捉座菠,并且在代碼產(chǎn)生異常后清理資源。當(dāng)然清理資源后藤树,可以繼續(xù)拋出非檢測(cè)異常浴滴,阻止程序的執(zhí)行。根據(jù)觀察和理解也榄,檢測(cè)異常大多可以應(yīng)用于工具類中巡莹。

誤區(qū)二司志、將異常直接顯示在頁面或客戶端甜紫。

將異常直接打印在客戶端的例子屢見不鮮,以 JSP 為例骂远,一旦代碼運(yùn)行出現(xiàn)異常囚霸,默認(rèn)情況下容器將異常堆棧信息直接打印在頁面上。其實(shí)從客戶角度來說激才,任何異常都沒有實(shí)際意義拓型,絕大多數(shù)的客戶也根本看不懂異常信息,軟件開發(fā)也要盡量避免將異常直接呈現(xiàn)給用戶瘸恼。

package com.ibm.dw.sample.exception;

/**

自定義 RuntimeException

添加錯(cuò)誤代碼屬性

*/

public class RuntimeException extends java.lang.RuntimeException {

//默認(rèn)錯(cuò)誤代碼

public static final Integer GENERIC = 1000000;

//錯(cuò)誤代碼

private Integer errorCode;

public RuntimeException(Integer errorCode, Throwable cause) {

this(errorCode, null, cause);

}

public RuntimeException(String message, Throwable cause) {

//利用通用錯(cuò)誤代碼

this(GENERIC, message, cause);

}

public RuntimeException(Integer errorCode, String message, Throwable cause) {

super(message, cause);

this.errorCode = errorCode;

}

public Integer getErrorCode() {

returnerrorCode;

}

}

正如示例代碼所示劣挫,在異常中引入錯(cuò)誤代碼,一旦出現(xiàn)異常东帅,我們只要將異常的錯(cuò)誤代碼呈現(xiàn)給用戶压固,或者將錯(cuò)誤代碼轉(zhuǎn)換成更通俗易懂的提示。其實(shí)這里的錯(cuò)誤代碼還包含另外一個(gè)功能靠闭,開發(fā)人員亦可以根據(jù)錯(cuò)誤代碼準(zhǔn)確的知道了發(fā)生了什么類型異常帐我。

誤區(qū)三、對(duì)代碼層次結(jié)構(gòu)的污染

我們經(jīng)常將代碼分 Service愧膀、Business Logic拦键、DAO 等不同的層次結(jié)構(gòu),DAO 層中會(huì)包含拋出異常的方法檩淋,如下所示:

public Customer retrieveCustomerById(Long id) throw SQLException {

//根據(jù) ID 查詢數(shù)據(jù)庫

}

上面這段代碼咋一看沒什么問題芬为,但是從設(shè)計(jì)耦合角度仔細(xì)考慮一下,這里的 SQLException 污染到了上層調(diào)用代碼蟀悦,調(diào)用層需要顯式的利用 try-catch 捕捉媚朦,或者向更上層次進(jìn)一步拋出。根據(jù)設(shè)計(jì)隔離原則熬芜,我們可以適當(dāng)修改成:

public Customer retrieveCustomerById(Long id) {

try{

//根據(jù) ID 查詢數(shù)據(jù)庫

}catch(SQLException e){

//利用非檢測(cè)異常封裝檢測(cè)異常莲镣,降低層次耦合

throw new RuntimeException(SQLErrorCode, e);

}finally{

//關(guān)閉連接,清理資源

}

}alert("Hello CSDN");

誤區(qū)四涎拉、忽略異常

如下異常處理只是將異常輸出到控制臺(tái)瑞侮,沒有任何意義的圆。而且這里出現(xiàn)了異常并沒有中斷程序,進(jìn)而調(diào)用代碼繼續(xù)執(zhí)行半火,導(dǎo)致更多的異常越妈。

public void retrieveObjectById(Long id){

try{

//..some code that throws SQLException

}catch(SQLException ex){

/*了解的人都知道,這里的異常打印毫無意義钮糖,僅僅是將錯(cuò)誤堆棧輸出到控制臺(tái)梅掠。

* 而在 Production 環(huán)境中,需要將錯(cuò)誤堆棧輸出到日志店归。

* 而且這里 catch 處理之后程序繼續(xù)執(zhí)行阎抒,會(huì)導(dǎo)致進(jìn)一步的問題*/

ex.printStacktrace();

}

}

可以重構(gòu)成:

public void retrieveObjectById(Long id){

try{

//..some code that throws SQLException

}

catch(SQLException ex){

throw new RuntimeException(“Exception in retieveObjectById”, ex);

}

finally{

//clean up resultset, statement, connection etc

}

}

這個(gè)誤區(qū)比較基本,一般情況下都不會(huì)犯此低級(jí)錯(cuò)誤消痛。

誤區(qū)五且叁、將異常包含在循環(huán)語句塊中

如下代碼所示,異常包含在 for 循環(huán)語句塊中秩伞。

for(int i=0; i<100; i++){

try{

}catch(XXXException e){

//….

}

}

我們都知道異常處理占用系統(tǒng)資源逞带。一看,大家都認(rèn)為不會(huì)犯這樣的錯(cuò)誤纱新。換個(gè)角度展氓,類 A 中執(zhí)行了一段循環(huán),循環(huán)中調(diào)用了 B 類的方法脸爱,B 類中被調(diào)用的方法卻又包含 try-catch 這樣的語句塊遇汞。褪去類的層次結(jié)構(gòu),代碼和上面如出一轍阅羹。

誤區(qū)六勺疼、利用 Exception 捕捉所有潛在的異常

一段方法執(zhí)行過程中拋出了幾個(gè)不同類型的異常,為了代碼簡潔捏鱼,利用基類 Exception 捕捉所有潛在的異常执庐,如下例所示:

public void retrieveObjectById(Long id){

try{

//…拋出 IOException 的代碼調(diào)用

//…拋出 SQLException 的代碼調(diào)用

}catch(Exception e){

//這里利用基類 Exception 捕捉的所有潛在的異常,如果多個(gè)層次這樣捕捉导梆,會(huì)丟失原始異常的有效信息

throw new RuntimeException(“Exception in retieveObjectById”, e);

}

}

可以重構(gòu)成

public void retrieveObjectById(Long id){

try{

//..some code that throws RuntimeException, IOException, SQLException

}catch(IOException e){

//僅僅捕捉 IOException

throw new RuntimeException(/指定這里 IOException 對(duì)應(yīng)的錯(cuò)誤代碼/code,“Exception in retieveObjectById”, e);

}catch(SQLException e){

//僅僅捕捉 SQLException

throw new RuntimeException(/指定這里 SQLException 對(duì)應(yīng)的錯(cuò)誤代碼/code,“Exception in retieveObjectById”, e);

}

}

誤區(qū)七轨淌、多層次封裝拋出非檢測(cè)異常

如果我們一直堅(jiān)持不同類型的異常一定用不同的捕捉語句,那大部分例子可以繞過這一節(jié)了看尼。但是如果僅僅一段代碼調(diào)用會(huì)拋出一種以上的異常時(shí)递鹉,很多時(shí)候沒有必要每個(gè)不同類型的 Exception 寫一段 catch 語句,對(duì)于開發(fā)來說藏斩,任何一種異常都足夠說明了程序的具體問題躏结。

try{

//可能拋出 RuntimeException、IOExeption 或者其它狰域;

//注意這里和誤區(qū)六的區(qū)別媳拴,這里是一段代碼拋出多種異常黄橘。以上是多段代碼,各自拋出不同的異常

}catch(Exception e){

//一如既往的將 Exception 轉(zhuǎn)換成 RuntimeException屈溉,但是這里的 e 其實(shí)是 RuntimeException 的實(shí)例塞关,已經(jīng)在前段代碼中封裝過

throw new RuntimeException(//code, //, e);

}

如果我們?nèi)缟侠荆瑢⑺械?Exception 再轉(zhuǎn)換成 RuntimeException子巾,那么當(dāng) Exception 的類型已經(jīng)是 RuntimeException 時(shí)帆赢,我們又做了一次封裝。將 RuntimeException 又重新封裝了一次线梗,進(jìn)而丟失了原有的 RuntimeException 攜帶的有效信息椰于。

解決辦法是我們可以在 RuntimeException 類中添加相關(guān)的檢查,確認(rèn)參數(shù) Throwable 不是 RuntimeException 的實(shí)例缠导。如果是廉羔,將拷貝相應(yīng)的屬性到新建的實(shí)例上溉痢∑г欤或者用不同的 catch 語句塊捕捉 RuntimeException 和其它的 Exception。個(gè)人偏好方式一孩饼,好處不言而喻髓削。

誤區(qū)八、多層次打印異常

我們先看一下下面的例子镀娶,定義了 2 個(gè)類 A 和 B立膛。其中 A 類中調(diào)用了 B 類的代碼,并且 A 類和 B 類中都捕捉打印了異常梯码。

public class A {

private static Logger logger = LoggerFactory.getLogger(A.class);

public void process(){

try{

//實(shí)例化 B 類宝泵,可以換成其它注入等方式

B b = new B();

b.process();

//other code might cause exception

} catch(XXXException e){

//如果 B 類 process 方法拋出異常,異常會(huì)在 B 類中被打印轩娶,在這里也會(huì)被打印儿奶,從而會(huì)打印 2 次

logger.error(e);

throw new RuntimeException(/錯(cuò)誤代碼/ errorCode, /異常信息/msg, e);

}

}

}

public class B{

private static Logger logger = LoggerFactory.getLogger(B.class);

public void process(){

try{

//可能拋出異常的代碼

}

catch(XXXException e){

logger.error(e);

throw new RuntimeException(/錯(cuò)誤代碼/ errorCode, /異常信息/msg, e);

}

}

}

同一段異常會(huì)被打印 2 次。如果層次再復(fù)雜一點(diǎn)鳄抒,不去考慮打印日志消耗的系統(tǒng)性能闯捎,僅僅在異常日志中去定位異常具體的問題已經(jīng)夠頭疼的了。

其實(shí)打印日志只需要在代碼的最外層捕捉打印就可以了许溅,異常打印也可以寫成 AOP瓤鼻,織入到框架的最外層。

誤區(qū)九贤重、異常包含的信息不能充分定位問題

異常不僅要能夠讓開發(fā)人員知道哪里出了問題茬祷,更多時(shí)候開發(fā)人員還需要知道是什么原因?qū)е碌膯栴},我們知道 java .lang.Exception 有字符串類型參數(shù)的構(gòu)造方法并蝗,這個(gè)字符串可以自定義成通俗易懂的提示信息祭犯。

簡單的自定義信息開發(fā)人員只能知道哪里出現(xiàn)了異常耐朴,但是很多的情況下,開發(fā)人員更需要知道是什么參數(shù)導(dǎo)致了這樣的異常盹憎。這個(gè)時(shí)候我們就需要將方法調(diào)用的參數(shù)信息追加到自定義信息中筛峭。下例只列舉了一個(gè)參數(shù)的情況,多個(gè)參數(shù)的情況下陪每,可以單獨(dú)寫一個(gè)工具類組織這樣的字符串影晓。

public void retieveObjectById(Long id){

try{

//..some code that throws SQLException

}catch(SQLException ex){

//將參數(shù)信息添加到異常信息中

throw new RuntimeException(“Exception in retieveObjectById with Object Id :”+ id, ex);

}

}

誤區(qū)十、不能預(yù)知潛在的異常

在寫代碼的過程中檩禾,由于對(duì)調(diào)用代碼缺乏深層次的了解挂签,不能準(zhǔn)確判斷是否調(diào)用的代碼會(huì)產(chǎn)生異常,因而忽略處理盼产。在產(chǎn)生了 Production Bug 之后才想起來應(yīng)該在某段代碼處添加異常補(bǔ)捉饵婆,甚至不能準(zhǔn)確指出出現(xiàn)異常的原因。這就需要開發(fā)人員不僅知道自己在做什么戏售,而且要去盡可能的知道別人做了什么侨核,可能會(huì)導(dǎo)致什么結(jié)果,從全局去考慮整個(gè)應(yīng)用程序的處理過程灌灾。這些思想會(huì)影響我們對(duì)代碼的編寫和處理搓译。

誤區(qū)十一、混用多種第三方日志庫

現(xiàn)如今 Java 第三方日志庫的種類越來越多锋喜,一個(gè)大項(xiàng)目中會(huì)引入各種各樣的框架些己,而這些框架又會(huì)依賴不同的日志庫的實(shí)現(xiàn)。最麻煩的問題倒不是引入所有需要的這些日志庫嘿般,問題在于引入的這些日志庫之間本身不兼容段标。如果在項(xiàng)目初期可能還好解決,可以把所有代碼中的日志庫根據(jù)需要重新引入一遍炉奴,或者換一套框架逼庞。但這樣的成本不是每個(gè)項(xiàng)目都承受的起的,而且越是隨著項(xiàng)目的進(jìn)行盆佣,這種風(fēng)險(xiǎn)就越大往堡。

怎么樣才能有效的避免類似的問題發(fā)生呢,現(xiàn)在的大多數(shù)框架已經(jīng)考慮到了類似的問題共耍,可以通過配置 Properties 或 xml 文件虑灰、參數(shù)或者運(yùn)行時(shí)掃描 Lib 庫中的日志實(shí)現(xiàn)類,真正在應(yīng)用程序運(yùn)行時(shí)才確定具體應(yīng)用哪個(gè)特定的日志庫痹兜。

其實(shí)根據(jù)不需要多層次打印日志那條原則穆咐,我們就可以簡化很多原本調(diào)用日志打印代碼的類。很多情況下,我們可以利用攔截器或者過濾器實(shí)現(xiàn)日志的打印对湃,降低代碼維護(hù)崖叫、遷移的成本。

結(jié)束語

以上純屬個(gè)人的經(jīng)驗(yàn)和總結(jié)拍柒,任何事物都是辯證的心傀,沒有絕對(duì)的原則,適合自己的才是最有效的原則拆讯。希望以上的講解和分析可以對(duì)您有所幫助脂男。想了解更多可以點(diǎn)擊加入QQ群:514790886 里面有Java高級(jí)大牛直播講解知識(shí)點(diǎn) 走的就是高端路線(如果你想跳槽換工作 但是技術(shù)又不夠 或者工作上遇到了瓶頸 我這里有一個(gè)JAVA的免費(fèi)直播課程 講的是高端的知識(shí)點(diǎn)基礎(chǔ)不好的勿入喲 只要你有1-5年的開發(fā)經(jīng)驗(yàn)可以加群找我要課堂鏈接 注意:是免費(fèi)的 沒有開發(fā)經(jīng)驗(yàn)勿入哦)

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市种呐,隨后出現(xiàn)的幾起案子宰翅,更是在濱河造成了極大的恐慌,老刑警劉巖爽室,帶你破解...
    沈念sama閱讀 217,277評(píng)論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件汁讼,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡阔墩,警方通過查閱死者的電腦和手機(jī)嘿架,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評(píng)論 3 393
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來戈擒,“玉大人眶明,你說我怎么就攤上這事】鸶撸” “怎么了?”我有些...
    開封第一講書人閱讀 163,624評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵丑瞧,是天一觀的道長柑土。 經(jīng)常有香客問我,道長绊汹,這世上最難降的妖魔是什么稽屏? 我笑而不...
    開封第一講書人閱讀 58,356評(píng)論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮西乖,結(jié)果婚禮上狐榔,老公的妹妹穿的比我還像新娘。我一直安慰自己获雕,他們只是感情好薄腻,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,402評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著届案,像睡著了一般庵楷。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,292評(píng)論 1 301
  • 那天尽纽,我揣著相機(jī)與錄音咐蚯,去河邊找鬼。 笑死弄贿,一個(gè)胖子當(dāng)著我的面吹牛春锋,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播差凹,決...
    沈念sama閱讀 40,135評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼看疙,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了直奋?” 一聲冷哼從身側(cè)響起能庆,我...
    開封第一講書人閱讀 38,992評(píng)論 0 275
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎脚线,沒想到半個(gè)月后搁胆,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,429評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡邮绿,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,636評(píng)論 3 334
  • 正文 我和宋清朗相戀三年渠旁,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片船逮。...
    茶點(diǎn)故事閱讀 39,785評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡顾腊,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出挖胃,到底是詐尸還是另有隱情杂靶,我是刑警寧澤,帶...
    沈念sama閱讀 35,492評(píng)論 5 345
  • 正文 年R本政府宣布酱鸭,位于F島的核電站吗垮,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏凹髓。R本人自食惡果不足惜烁登,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,092評(píng)論 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望蔚舀。 院中可真熱鬧饵沧,春花似錦、人聲如沸赌躺。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,723評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽寿谴。三九已至锁右,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背咏瑟。 一陣腳步聲響...
    開封第一講書人閱讀 32,858評(píng)論 1 269
  • 我被黑心中介騙來泰國打工拂到, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人码泞。 一個(gè)月前我還...
    沈念sama閱讀 47,891評(píng)論 2 370
  • 正文 我出身青樓兄旬,卻偏偏與公主長得像,于是被迫代替她去往敵國和親余寥。 傳聞我的和親對(duì)象是個(gè)殘疾皇子领铐,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,713評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容

  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語法,類相關(guān)的語法宋舷,內(nèi)部類的語法绪撵,繼承相關(guān)的語法,異常的語法祝蝠,線程的語...
    子非魚_t_閱讀 31,625評(píng)論 18 399
  • 人的一生中總要遇到迷茫音诈,這迷茫可能是自己的主觀绎狭,也有可能是環(huán)境造成细溅。 誰的生活不迷茫, 不迷茫的生活怎有味...
    孤度閱讀 228評(píng)論 0 3
  • 鳳娘儡嘶,就是鳳的娘喇聊。沒有人知道她叫什么,因?yàn)樗袀€(gè)女兒叫鳳蹦狂,所以她就是鳳她娘誓篱,后來就叫成了鳳娘。 鳳娘不是這個(gè)村子的...
    非錦玉閱讀 2,498評(píng)論 13 19
  • 測(cè)評(píng) 帖子
    home90閱讀 176評(píng)論 0 0