誤區(qū)七、多層次封裝拋出非檢測(cè)異常
如果我們一直堅(jiān)持不同類型的異常一定用不同的捕捉語句,那大部分例子可以繞過這一節(jié)了。但是如果僅僅一段代碼調(diào)用會(huì)拋出一種以上的異常時(shí),很多時(shí)候沒有必要每個(gè)不同類型的 Exception 寫一段 catch 語句,對(duì)于開發(fā)來說,任何一種異常都足夠說明了程序的具體問題。
清單 9
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 類中都捕捉打印了異常。
清單 10
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è)字符串可以自定義成通俗易懂的提示信息。
簡(jiǎn)單的自定義信息開發(fā)人員只能知道哪里出現(xiàn)了異常,但是很多的情況下,開發(fā)人員更需要知道是什么參數(shù)導(dǎo)致了這樣的異常。這個(gè)時(shí)候我們就需要將方法調(diào)用的參數(shù)信息追加到自定義信息中。下例只列舉了一個(gè)參數(shù)的情況,多個(gè)參數(shù)的情況下,可以單獨(dú)寫一個(gè)工具類組織這樣的字符串。
清單 11
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ù)不需要多層次打印日志那條原則,我們就可以簡(jiǎn)化很多原本調(diào)用日志打印代碼的類。很多情況下,我們可以利用攔截器或者過濾器實(shí)現(xiàn)日志的打印,降低代碼維護(hù)、遷移的成本。
結(jié)束語
以上純來自網(wǎng)友的經(jīng)驗(yàn)和總結(jié),事物都是辯證的,沒有絕對(duì)的原則,適合自己的才是最有效的原則。希望以上的講解和分析可以對(duì)您有所幫助。
評(píng)論
查看更多