結論:
- 不管有沒有出現異常,finally代碼塊都會執行;
- 當try和catch中有return語句時,finally仍然會執行;
- finally是在return後面的表達式運算後執行的(此時并沒有傳回運算後的值,而是先把要傳回的值儲存起來,管finally中的代碼怎麼樣,傳回的值都不會改變,任然是之前儲存的值),是以函數傳回值是在finally執行前确定的;
- 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進行了改變,但是不會影響傳回結果。
它應該使用棧儲存傳回值。