天天看點

有return的情況下try catch finally的執行順序的全面總結

結論:

  1. 不管有沒有出現異常,finally代碼塊都會執行;
  2. 當try和catch中有return語句時,finally仍然會執行;
  3. finally是在return後面的表達式運算後執行的(此時并沒有傳回運算後的值,而是先把要傳回的值儲存起來,管finally中的代碼怎麼樣,傳回的值都不會改變,任然是之前儲存的值),是以函數傳回值是在finally執行前确定的;
  4. finally中最好不要包含return語句,否則程式會提前退出,傳回值不是try或catch中儲存的傳回值。

舉例:

情況1::try{} catch(){}finally{} return;

顯然程式按順序執行。

情況2: try{ return; }catch(){} finally{} return;

程式執行try塊中return之前(包括return語句中的表達式運算)代碼;

再執行finally塊,最後執行try中return;(最後一個return語句不執行)

情況3: try{ } catch(){return;} finally{} return;

程式先執行try,若:

有異常:則執行catch中return之前(包括return語句中的表達式運算)代碼,再執行finally語句中全部代碼,最後執行catch塊中return。 finally後的return不再執行。

無異常:執行完try再finally再執行最後一個return.

情況4: try{ return; }catch(){} finally{return;}

程式先執行try塊中return之前(包括return語句中的表達式運算)代碼;

再執行finally塊,因為finally塊中有return是以提前退出。

情況5: try{} catch(){return;}finally{return;}

與情況4類似,因為finally塊中有return是以提前退出。

情況6: try{ return;}catch(){return;} finally{return;}

程式執行try塊中return之前(包括return語句中的表達式運算)代碼;

有異常:執行catch塊中return之前(包括return語句中的表達式運算)代碼;

則再執行finally塊,因為finally塊中有return是以提前退出。

無異常:則再執行finally塊,因為finally塊中有return是以提前退出。

最終結論:任何執行try 或者catch中的return語句之前,(如果有finally)都會先執行finally語句。

如果finally中有return語句,那麼程式就return了,是以finally中的return是一定會被return的,

編譯器把finally中的return實作為一個warning。

測試程式:

public class Test  
{
    public static void main(String[] args) {

        System.out.println(test());;
    }

    public static int test()
    {
        int x = ;
        try
        {
            x++;
            return x;
        }
        finally
        {
            ++x;
        }
    }
}
           

結果是2。

分析:

在try語句中,在執行return語句時,要傳回的結果已經準備好了,就在此時,程式轉到finally執行了。

在轉去之前,try中先把要傳回的結果存放到不同于x的局部變量中去,執行完finally之後,在從中取出傳回結果,是以,即使finally中對變量x進行了改變,但是不會影響傳回結果。

它應該使用棧儲存傳回值。