Markdown版本筆記 | 我的GitHub首頁 | 我的博客 | 我的微信 | 我的郵箱 |
---|---|---|---|---|
MyAndroidBlogs | baiqiantao | baiqiantao | bqt20094 | baiqiantao@sina.com |
異常 try catch finally return 執行關系 MD
目錄
探討finally語句的執行與return的關系
探討finally語句的執行與return的關系
Java異常捕獲機制try...catch...finally
塊中的finally
語句是不是一定會被執行?不一定,至少有兩種情況下finally語句是不會被執行的:
- try語句沒有被執行到,如在try語句之前就返回了,這樣finally語句就不會執行,這也說明了finally語句被執行的必要而非充分條件是:相應的try語句一定被執行到。
- 在try塊中有
System.exit(0)
這樣的語句,System.exit(0)
是終止Java虛擬機JVM的,連JVM都停止了,所有都結束了,當然finally語句也不會被執行到。
1、finally語句是在try的return語句執行之后,return返回之前執行的
測試案例:
public class TestFinally {public static void main(String[] args) {System.out.println(test());}public static String test() {try {System.out.println("try block");if (new Random().nextBoolean()) {return "直接返回";} else {return test2();}} finally {System.out.println("finally block");}}public static String test2() {System.out.println("return statement");return "調用方法返回";}
}
運行結果:
try block
finally block
直接返回
或
try block
return statement
finally block
調用方法返回
說明try中的return語句先執行了,但并沒有立即返回,而是等到finally執行結束后再返回。
這里大家可能會想:如果finally里也有return語句,那么是不是就直接返回了,try中的return就不能返回了?看下面。
2、finally塊中的return語句會覆蓋try塊中的return返回
public class TestFinally {public static void main(String[] args) {System.out.println(test());}public static String test() {try {System.out.println("try block");return "在try中返回";} finally {System.out.println("finally block");return "在finally中返回";}// return "finally外面的return就變成不可到達語句,需要注釋掉否則編譯器報錯";}
}
運行結果:
try block
finally block
在finally中返回
這說明finally里的return直接返回了,就不管try中是否還有返回語句。
這里還有個小細節需要注意,finally里加上return過后,finally外面的return b就變成不可到達語句了,也就是永遠不能被執行到,所以需要注釋掉否則編譯器報錯。
3、如果finally語句中沒有return語句覆蓋返回值,那么原來的返回值可能因為finally里的修改而改變,也可能不變
測試用例:
public class TestFinally {public static void main(String[] args) {System.out.println(test());}public static int test() {int b = 20;try {System.out.println("try block");return b += 80;} finally {b += 10;System.out.println("finally block");}}
}
運行結果:
try block
finally block
100
測試用例2:
public class TestFinally {public static void main(String[] args) {System.out.println(test());}public static List<Integer> test() {List<Integer> list = new ArrayList<Integer>();list.add(10086);try {System.out.println("try block");return list;} finally {list.add(10088);System.out.println("finally block");}}
}
運行結果:
try block
finally block
[10086, 10088]
這其實就是Java到底是傳值還是傳址的問題了,簡單來說就是:Java中只有傳值沒有傳址。
這里大家可能又要想:是不是每次返回的一定是try中的return語句呢?那么finally外的return不是一點作用沒嗎?請看下面
4、try塊里的return語句在異常的情況下不會被執行,這樣具體返回哪個看情況
public class TestFinally {public static void main(String[] args) {System.out.println(test());}public static int test() {int b = 0;try {System.out.println("try block");b = b / 0;return b += 1;} catch (Exception e) {b += 10;System.out.println("catch block");} finally {b += 100;System.out.println("finally block");}return b;}
}
運行結果是:
try block
catch block
finally block
110
這里因 為在return之前發生了異常,所以try中的return不會被執行到
,而是接著執行捕獲異常的 catch 語句和最終的 finally 語句,此時兩者對b的修改都影響了最終的返回值,這時最后的 return b 就起到作用了。
這里大家可能又有疑問:如果catch中有return語句呢?當然只有在異常的情況下才有可能會執行,那么是在 finally 之前就返回嗎?看下面。
5、當發生異常后,catch中的return執行情況與未發生異常時try中return的執行情況完全一樣
public class TestFinally {public static void main(String[] args) {System.out.println(test());}public static int test() {int b = 0;try {System.out.println("try block");b = b / 0;return b += 1;} catch (Exception e) {b += 10;System.out.println("catch block");return 10086;} finally {b += 100;System.out.println("finally block");}//return b;}
}
運行結果:
try block
catch block
finally block
10086
說明了發生異常后,catch中的return語句先執行,確定了返回值后再去執行finally塊,執行完了catch再返回,也就是說情況與try中的return語句執行完全一樣。
總結:
- finally塊的語句在try或catch中的return語句執行之后返回之前執行
- 且finally里的修改語句可能影響也可能不影響try或catch中return已經確定的返回值
- 若finally里也有return語句則覆蓋try或catch中的return語句直接返回
2019-3-4