本文著重介紹了 Java 異常選擇和使用中的一些誤區(qū),希望各位讀者能夠熟練掌握異常處理的一些注意點(diǎn)和原則,注意總結(jié)和歸納。只有處理好了異常,才能提升開發(fā)人員的基本素養(yǎng),提高系統(tǒng)的健壯性,提升用戶體驗(yàn),提高產(chǎn)品的價值。
誤區(qū)一、異常的選擇
圖 1. 異常分類
圖 1 描述了異常的結(jié)構(gòu),其實(shí)我們都知道異常分檢測異常和非檢測異常,但是在實(shí)際中又混淆了這兩種異常的應(yīng)用。由于非檢測異常使用方便,很多開發(fā)人員就認(rèn)為檢測異常沒什么用處。其實(shí)異常的應(yīng)用情景可以概括為以下:
一、調(diào)用代碼不能繼續(xù)執(zhí)行,需要立即終止。出現(xiàn)這種情況的可能性太多太多,例如服務(wù)器連接不上、參數(shù)不正確等。這些時候都適用非檢測異常,不需要調(diào)用代碼的顯式捕捉和處理,而且代碼簡潔明了。
二、調(diào)用代碼需要進(jìn)一步處理和恢復(fù)。假如將 SQLException 定義為非檢測異常,這樣操作數(shù)據(jù)時開發(fā)人員理所當(dāng)然的認(rèn)為 SQLException 不需要調(diào)用代碼的顯式捕捉和處理,進(jìn)而會導(dǎo)致嚴(yán)重的 Connection 不關(guān)閉、Transaction 不回滾、DB 中出現(xiàn)臟數(shù)據(jù)等情況,正因?yàn)?SQLException 定義為檢測異常,才會驅(qū)使開發(fā)人員去顯式捕捉,并且在代碼產(chǎn)生異常后清理資源。當(dāng)然清理資源后,可以繼續(xù)拋出非檢測異常,阻止程序的執(zhí)行。根據(jù)觀察和理解,檢測異常大多可以應(yīng)用于工具類中。
誤區(qū)二、將異常直接顯示在頁面或客戶端。
將異常直接打印在客戶端的例子屢見不鮮,以 JSP 為例,一旦代碼運(yùn)行出現(xiàn)異常,默認(rèn)情況下容器將異常堆棧信息直接打印在頁面上。其實(shí)從客戶角度來說,任何異常都沒有實(shí)際意義,絕大多數(shù)的客戶也根本看不懂異常信息,軟件開發(fā)也要盡量避免將異常直接呈現(xiàn)給用戶。
清單 1
package com.ibm.dw.sample.exception;
/**
* 自定義 RuntimeException
* 添加錯誤代碼屬性
*/
public class RuntimeException extends java.lang.RuntimeException {
//默認(rèn)錯誤代碼
public static final Integer GENERIC = 1000000;
//錯誤代碼
private Integer errorCode;
public RuntimeException(Integer errorCode, Throwable cause) {
this(errorCode, null, cause);
}
public RuntimeException(String message, Throwable cause) {
//利用通用錯誤代碼
this(GENERIC, message, cause);
}
public RuntimeException(Integer errorCode, String message, Throwable cause) {
super(message, cause);
this.errorCode = errorCode;
}
public Integer getErrorCode() {
return errorCode;
}
}
正如示例代碼所示,在異常中引入錯誤代碼,一旦出現(xiàn)異常,我們只要將異常的錯誤代碼呈現(xiàn)給用戶,或者將錯誤代碼轉(zhuǎn)換成更通俗易懂的提示。其實(shí)這里的錯誤代碼還包含另外一個功能,開發(fā)人員亦可以根據(jù)錯誤代碼準(zhǔn)確的知道了發(fā)生了什么類型異常。
誤區(qū)三、對代碼層次結(jié)構(gòu)的污染
我們經(jīng)常將代碼分 Service、Business Logic、DAO 等不同的層次結(jié)構(gòu),DAO 層中會包含拋出異常的方法,如清單 2 所示:
清單 2
public Customer retrieveCustomerById(Long id) throw SQLException {
//根據(jù) ID 查詢數(shù)據(jù)庫
}
上面這段代碼咋一看沒什么問題,但是從設(shè)計耦合角度仔細(xì)考慮一下,這里的 SQLException 污染到了上層調(diào)用代碼,調(diào)用層需要顯式的利用 try-catch 捕捉,或者向更上層次進(jìn)一步拋出。根據(jù)設(shè)計隔離原則,我們可以適當(dāng)修改成:
清單 3
public Customer retrieveCustomerById(Long id) {
try{
//根據(jù) ID 查詢數(shù)據(jù)庫
}catch(SQLException e){
//利用非檢測異常封裝檢測異常,降低層次耦合
throw new RuntimeException(SQLErrorCode, e);
}finally{
//關(guān)閉連接,清理資源
}
}
評論
查看更多