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語句呢?try和catch內的return會丟失,實際會返回finally中的返回值。finally中有return不僅會覆蓋try和catch內的返回值,還會掩蓋try和catch內的異常,就像異常沒有發生一樣,比如說:
public static int test(){ int ret = 0; try{ int a = 5/0; return ret; }finally{ return 2; } }
以上代碼中,5/0會觸發ArithmeticException,但是finally中有return語句,這個方法就會返回2,而不再向上傳遞異常了。
finally中不僅return語句會掩蓋異常,如果finally中拋出了異常,則原異常就會被掩蓋,看下面代碼:
public static void test(){ try{ int a = 5/0; }finally{ throw new RuntimeException("hello"); } }
finally中拋出了RuntimeException,則原異常ArithmeticException就丟失了。
所以,一般而言,為避免混淆,應該避免在finally中使用return語句或者拋出異常,如果調用的其他代碼可能拋出異常,則應該捕獲異常並進行處理。