天天看點

異常 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語句呢?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語句或者抛出異常,如果調用的其他代碼可能抛出異常,則應該捕獲異常并進行處理。