用法
1、將預見可能引發異常的代碼包含在try語句塊中。
2、如果發生了異常,則轉入catch的執行。
catch有幾種寫法:
catch 這將捕獲任何發生的異常。
catch(Exception e) 這將捕獲任何發生的異常。另外,還提供e參數,你可以在處理異常時使用e參數來獲得有關異常的信息。
catch(Exception的派生類 e) 這將捕獲派生類定義的異常,例如安卓中文網,我想捕獲一個無效操作的異常,可以如下寫:
catch(InvalidOperationException e) { .... } 這樣,如果try語句塊中拋出的異常是InvalidOperationException,將轉入該處執行,其他異常不處理。
catch可以有多個,也可以沒有,每個catch可以處理一個特定的異常。.net按照你catch的順序查找異常處理塊,如果找到,則進行處理,如果找不到,則向上一層次拋出。如果沒有上一層次,則向用戶拋出,此時,如果你在調試,程序將中斷運行,如果是部署的程序,將會中止。 如果沒有catch塊,異常總是向上層(如果有)拋出,或者中斷程序運行。
3、finally
finally可以沒有,也可以只有一個。無論有沒有發生異常,它總會在這個異常處理結構的最後運行。即使你在try塊內用return返回了,在返回前,finally總是要執行,這以便讓你有機會能夠在異常處理最後做一些清理工作。如關閉數據庫連接等等。
注意:如果沒有catch語句塊,那麼finally塊就是必須的。 如果你不希望在這裏處理異常,而當異常發生時提交到上層處理,但在這個地方無論發生異常,都要必須要執行一些操作,就可以使用try finally, 很典型的應用就是進行數據庫操作
finally
異常機制中還有一個重要的部分,就是finally, catch後面可以跟finally語句,語法如下所示:
try{
//可能拋出異常
}catch(Exception e){
//捕獲異常
}finally{
//不管有無異常都執行
}
finally內的代碼不管有無異常發生,都會執行。具體來說:
- 如果沒有異常發生,在try內的代碼執行結束後執行。
- 如果有異常發生且被catch捕獲,在catch內的代碼執行結束後執行
- 如果有異常發生但沒被捕獲,則在異常被拋給上層之前執行。
由於finally的這個特點,它一般用於釋放資源,如數據庫連接、文件流等。
try/catch/finally語法中,catch不是必需的,也就是可以只有try/finally,表示不捕獲異常,異常自動向上傳遞,但finally中的代碼在異常發生後也執行。
finally語句有一個執行細節,如果在try或者catch語句內有return語句,則return語句在finally語句執行結束後才執行,但finally並不能改變返回值,我們來看下代碼:
public static int test(){
int ret = 0;
try{
return ret;
}finally{
ret = 2;
}
}
這個函數的返回值是0,而不是2,實際執行過程是,在執行到try內的return ret;語句前,會先將返回值ret保存在一個臨時變量中,然後才執行finally語句,最後try再返回那個臨時變量,finally中對ret的修改不會被返回。
如果在finally中也有return語句呢?
編譯時會提示 “控制不能離開Finally子句主體。”
如果在Finally中拋出異常的話,會將try和catch中拋出的異常全部給掩蓋。
public static void test(){
try{
int a = 5/0;
}finally{
throw new RuntimeException("hello");
}
}
finally中拋出了RuntimeException,則原異常ArithmeticException就丟失了。
所以,一般而言,爲避免混淆,應該避免在finally中使用return語句或者拋出異常,如果調用的其他代碼可能拋出異常,則應該捕獲異常並進行處理。