異常 try – finally 注意的地方

finally數據庫

異常機制中還有一個重要的部分,就是finally, catch後面能夠跟finally語句,語法以下所示:
函數

 
try{
    //可能拋出異常
}catch(Exception e){
    //捕獲異常
}finally{
    //無論有無異常都執行
}
 

finally內的代碼無論有無異常發生,都會執行。具體來講:code

  • 若是沒有異常發生,在try內的代碼執行結束後執行。
  • 若是有異常發生且被catch捕獲,在catch內的代碼執行結束後執行
  • 若是有異常發生但沒被捕獲,則在異常被拋給上層以前執行。

因爲finally的這個特色,它通常用於釋放資源,如數據庫鏈接、文件流等。blog

try/catch/finally語法中,catch不是必需的,也就是能夠只有try/finally,表示不捕獲異常,異常自動向上傳遞,但finally中的代碼在異常發生後也執行。資源

finally語句有一個執行細節,若是在try或者catch語句內有return語句,則return語句在finally語句執行結束後才執行,但finally並不能改變返回值,咱們來看下代碼:it

 
public static int test(){
    int ret = 0;
    try{
        return ret;
    }finally{
        ret = 2;
    }
}
 

這個函數的返回值是0,而不是2,實際執行過程是,在執行到try內的return ret;語句前,會先將返回值ret保存在一個臨時變量中,而後才執行finally語句,最後try再返回那個臨時變量,finally中對ret的修改不會被返回。io

若是在finally中也有return語句呢?try和catch內的return會丟失,實際會返回finally中的返回值。finally中有return不只會覆蓋try和catch內的返回值,還會掩蓋try和catch內的異常,就像異常沒有發生同樣,好比說:class

 
public static int test(){
    int ret = 0;
    try{
        int a = 5/0;
        return ret;
    }finally{
        return 2;
    }
}
 

以上代碼中,5/0會觸發ArithmeticException,可是finally中有return語句,這個方法就會返回2,而再也不向上傳遞異常了。test

finally中不只return語句會掩蓋異常,若是finally中拋出了異常,則原異常就會被掩蓋,看下面代碼:變量

 
public static void test(){
    try{
        int a = 5/0;
    }finally{
        throw new RuntimeException("hello");
    }
}
 

finally中拋出了RuntimeException,則原異常ArithmeticException就丟失了。

因此,通常而言,爲避免混淆,應該避免在finally中使用return語句或者拋出異常,若是調用的其餘代碼可能拋出異常,則應該捕獲異常並進行處理。

相關文章
相關標籤/搜索