Java異常處理的11大誤區(qū)及經(jīng)驗(yàn)總結(jié)

在寫代碼的過程中,我們往往會(huì)忽略一些異常處理的基礎(chǔ)知識(shí)。本文將著重介紹 Java 異常選擇和使用中的一些誤區(qū),希望各位讀者能夠熟練掌握異常處理的一些注意點(diǎn)和原則,注意總結(jié)和歸納姨俩。

文章重點(diǎn):Java 異常的常見誤區(qū)和一些細(xì)節(jié)處理,包括異常的選擇师郑、錯(cuò)誤代碼的利用环葵、處理多層次的異常、以及如何添加有效信息到異常等宝冕。只有處理好了異常州既,才能提升開發(fā)人員的基本素養(yǎng)访娶,提高系統(tǒng)的健壯性朋沮,提升用戶體驗(yàn)毯盈,提高產(chǎn)品的價(jià)值。

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

圖 1. 異常分類

Java學(xué)習(xí)群宝剖,479121291洁闰,每天都會(huì)有干貨分享

圖 1 描述了異常的結(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)用代碼的顯式捕捉和處理砰诵,而且代碼簡(jiǎn)潔明了征唬。

在這里我推薦下我自己建的Java學(xué)習(xí)君羊:479121291,歡迎初學(xué)和進(jìn)階中的Java小伙伴進(jìn)入學(xué)習(xí)交流茁彭。

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ū)二:將異常直接顯示在頁(yè)面或客戶端

將異常直接打印在客戶端的例子屢見不鮮硝清,以 JSP 為例辅斟,一旦代碼運(yùn)行出現(xiàn)異常,默認(rèn)情況下容器將異常堆棧信息直接打印在頁(yè)面上耍缴。其實(shí)從客戶角度來(lái)說(shuō)砾肺,任何異常都沒有實(shí)際意義,絕大多數(shù)的客戶也根本看不懂異常信息防嗡,軟件開發(fā)也要盡量避免將異常直接呈現(xiàn)給用戶变汪。

清單 1

正如示例代碼所示,在異常中引入錯(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ì)包含拋出異常的方法淆党,如清單 2 所示:

清單 2

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

清單 3

誤區(qū)四:忽略異常

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

清單 4

可以重構(gòu)成:

清單 5

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

誤區(qū)五:將異常包含在循環(huán)語(yǔ)句塊中

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

清單 6

我們都知道異常處理占用系統(tǒng)資源玩荠。一看漆腌,大家都認(rèn)為不會(huì)犯這樣的錯(cuò)誤。換個(gè)角度阶冈,類 A 中執(zhí)行了一段循環(huán)闷尿,循環(huán)中調(diào)用了 B 類的方法,B 類中被調(diào)用的方法卻又包含 try-catch 這樣的語(yǔ)句塊女坑。褪去類的層次結(jié)構(gòu)填具,代碼和上面如出一轍。

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

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

清單 7

可以重構(gòu)成

清單 8

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

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

清單 9

如果我們?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 語(yǔ)句塊捕捉 RuntimeException 和其它的 Exception爆存。個(gè)人偏好方式一蛉顽,好處不言而喻。

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

我們先看一下下面的例子先较,定義了 2 個(gè)類 A 和 B携冤。其中 A 類中調(diào)用了 B 類的代碼悼粮,并且 A 類和 B 類中都捕捉打印了異常。

清單 10

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

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

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

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

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

清單 11

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

在寫代碼的過程中比搭,由于對(duì)調(diào)用代碼缺乏深層次的了解冠跷,不能準(zhǔn)確判斷是否調(diào)用的代碼會(huì)產(chǎn)生異常,因而忽略處理身诺。在產(chǎn)生了 Production Bug 之后才想起來(lái)應(yīng)該在某段代碼處添加異常補(bǔ)捉蜜托,甚至不能準(zhǔn)確指出出現(xiàn)異常的原因。這就需要開發(fā)人員不僅知道自己在做什么霉赡,而且要去盡可能的知道別人做了什么橄务,可能會(huì)導(dǎo)致什么結(jié)果,從全局去考慮整個(gè)應(yīng)用程序的處理過程穴亏。這些思想會(huì)影響我們對(duì)代碼的編寫和處理蜂挪。

最后我推薦下我自己的JAVAqun:479121291,不管你是小白還是大牛嗓化,小編我都挺歡迎棠涮,不定期分享干貨,包括我自己整理的一份最新JAVA資料和零基礎(chǔ)入門教程刺覆!严肪,歡迎初學(xué)和進(jìn)階中的小伙伴。

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

現(xiàn)如今 Java 第三方日志庫(kù)的種類越來(lái)越多,一個(gè)大項(xiàng)目中會(huì)引入各種各樣的框架驳糯,而這些框架又會(huì)依賴不同的日志庫(kù)的實(shí)現(xiàn)篇梭。最麻煩的問題倒不是引入所有需要的這些日志庫(kù),問題在于引入的這些日志庫(kù)之間本身不兼容酝枢。如果在項(xiàng)目初期可能還好解決恬偷,可以把所有代碼中的日志庫(kù)根據(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 庫(kù)中的日志實(shí)現(xiàn)類孕暇,真正在應(yīng)用程序運(yùn)行時(shí)才確定具體應(yīng)用哪個(gè)特定的日志庫(kù)。

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

結(jié)束語(yǔ)

以上純屬個(gè)人的經(jīng)驗(yàn)和總結(jié)陨帆,事物都是辯證的曲秉,沒有絕對(duì)的原則,適合自己的才是最有效的原則疲牵。希望以上的講解和分析可以對(duì)您有所幫助承二。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市纲爸,隨后出現(xiàn)的幾起案子亥鸠,更是在濱河造成了極大的恐慌,老刑警劉巖识啦,帶你破解...
    沈念sama閱讀 211,817評(píng)論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件负蚊,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡颓哮,警方通過查閱死者的電腦和手機(jī)家妆,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,329評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)冕茅,“玉大人伤极,你說(shuō)我怎么就攤上這事腰鬼。” “怎么了塑荒?”我有些...
    開封第一講書人閱讀 157,354評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)姜挺。 經(jīng)常有香客問我齿税,道長(zhǎng),這世上最難降的妖魔是什么炊豪? 我笑而不...
    開封第一講書人閱讀 56,498評(píng)論 1 284
  • 正文 為了忘掉前任凌箕,我火速辦了婚禮,結(jié)果婚禮上词渤,老公的妹妹穿的比我還像新娘牵舱。我一直安慰自己,他們只是感情好缺虐,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,600評(píng)論 6 386
  • 文/花漫 我一把揭開白布芜壁。 她就那樣靜靜地躺著,像睡著了一般高氮。 火紅的嫁衣襯著肌膚如雪慧妄。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,829評(píng)論 1 290
  • 那天剪芍,我揣著相機(jī)與錄音塞淹,去河邊找鬼。 笑死罪裹,一個(gè)胖子當(dāng)著我的面吹牛饱普,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播状共,決...
    沈念sama閱讀 38,979評(píng)論 3 408
  • 文/蒼蘭香墨 我猛地睜開眼套耕,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了口芍?” 一聲冷哼從身側(cè)響起箍铲,我...
    開封第一講書人閱讀 37,722評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎鬓椭,沒想到半個(gè)月后颠猴,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,189評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡小染,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,519評(píng)論 2 327
  • 正文 我和宋清朗相戀三年翘瓮,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片裤翩。...
    茶點(diǎn)故事閱讀 38,654評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡资盅,死狀恐怖调榄,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情呵扛,我是刑警寧澤每庆,帶...
    沈念sama閱讀 34,329評(píng)論 4 330
  • 正文 年R本政府宣布,位于F島的核電站今穿,受9級(jí)特大地震影響缤灵,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜蓝晒,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,940評(píng)論 3 313
  • 文/蒙蒙 一腮出、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧芝薇,春花似錦胚嘲、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,762評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至灭红,卻和暖如春侣滩,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背变擒。 一陣腳步聲響...
    開封第一講書人閱讀 31,993評(píng)論 1 266
  • 我被黑心中介騙來(lái)泰國(guó)打工君珠, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人娇斑。 一個(gè)月前我還...
    沈念sama閱讀 46,382評(píng)論 2 360
  • 正文 我出身青樓策添,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親毫缆。 傳聞我的和親對(duì)象是個(gè)殘疾皇子唯竹,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,543評(píng)論 2 349

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

  • 這篇文章主要是對(duì)Java異常選擇和使用中的一些誤區(qū)的總結(jié)和歸納,希望各位讀者能夠熟練掌握異常處理的一些注意點(diǎn)和原則...
    唐老鴨z閱讀 417評(píng)論 0 0
  • 最近的寫代碼的過程中苦丁,遇到很多異常的處理浸颓,以前上大學(xué)的時(shí)候?qū)懘a,遇到異常直接給個(gè)try catch了事旺拉,只是停留...
    德彪閱讀 899評(píng)論 0 1
  • 昨天晚上产上,由于未達(dá)成目標(biāo),我獨(dú)自做在自己的房間里蛾狗,房門敞開著晋涣,外面一片漆黑,每當(dāng)我向后面望去時(shí)沉桌,總會(huì)打幾個(gè)寒顫...
    樊4閱讀 300評(píng)論 0 0
  • 作者:爐叔 來(lái)源:圍爐夜讀(weiluyedu_) 世上有一種悲哀叫山無(wú)窮水無(wú)盡,話到嘴邊都說(shuō)盡佃扼。 一個(gè)星期前剛加...
    甜甜小果子閱讀 654評(píng)論 0 0
  • 不同的星座擁有不同的性格偎巢,今天我們來(lái)盤點(diǎn)一下最不喜歡被愛情束縛的星座男: 射手座: 射手座天生向往冒險(xiǎn),他們不喜歡...
    Random_ce08閱讀 294評(píng)論 0 0