Markdown版本笔记 我的GitHub首页 我的博客 我的微信 我的邮箱
MyAndroidBlogs baiqiantao baiqiantao bqt20094 baiqiantao@sina.com

异常 try catch finally return 执行关系 MD


目录

探讨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返回之前执行的

测试案例:

  1. public class TestFinally {
  2. public static void main(String[] args) {
  3. System.out.println(test());
  4. }
  5. public static String test() {
  6. try {
  7. System.out.println("try block");
  8. if (new Random().nextBoolean()) {
  9. return "直接返回";
  10. } else {
  11. return test2();
  12. }
  13. } finally {
  14. System.out.println("finally block");
  15. }
  16. }
  17. public static String test2() {
  18. System.out.println("return statement");
  19. return "调用方法返回";
  20. }
  21. }

运行结果:

  1. try block
  2. finally block
  3. 直接返回

  1. try block
  2. return statement
  3. finally block
  4. 调用方法返回

说明try中的return语句先执行了,但并没有立即返回,而是等到finally执行结束后再返回。

这里大家可能会想:如果finally里也有return语句,那么是不是就直接返回了,try中的return就不能返回了?看下面。

2、finally块中的return语句会覆盖try块中的return返回

  1. public class TestFinally {
  2. public static void main(String[] args) {
  3. System.out.println(test());
  4. }
  5. public static String test() {
  6. try {
  7. System.out.println("try block");
  8. return "在try中返回";
  9. } finally {
  10. System.out.println("finally block");
  11. return "在finally中返回";
  12. }
  13. // return "finally外面的return就变成不可到达语句,需要注释掉否则编译器报错";
  14. }
  15. }

运行结果:

  1. try block
  2. finally block
  3. finally中返回

这说明finally里的return直接返回了,就不管try中是否还有返回语句。

这里还有个小细节需要注意,finally里加上return过后,finally外面的return b就变成不可到达语句了,也就是永远不能被执行到,所以需要注释掉否则编译器报错。

3、如果finally语句中没有return语句覆盖返回值,那么原来的返回值可能因为finally里的修改而改变,也可能不变

测试用例:

  1. public class TestFinally {
  2. public static void main(String[] args) {
  3. System.out.println(test());
  4. }
  5. public static int test() {
  6. int b = 20;
  7. try {
  8. System.out.println("try block");
  9. return b += 80;
  10. } finally {
  11. b += 10;
  12. System.out.println("finally block");
  13. }
  14. }
  15. }

运行结果:

  1. try block
  2. finally block
  3. 100

测试用例2:

  1. public class TestFinally {
  2. public static void main(String[] args) {
  3. System.out.println(test());
  4. }
  5. public static List<Integer> test() {
  6. List<Integer> list = new ArrayList<Integer>();
  7. list.add(10086);
  8. try {
  9. System.out.println("try block");
  10. return list;
  11. } finally {
  12. list.add(10088);
  13. System.out.println("finally block");
  14. }
  15. }
  16. }

运行结果:

  1. try block
  2. finally block
  3. [10086, 10088]

这其实就是Java到底是传值还是传址的问题了,简单来说就是:Java中只有传值没有传址。

这里大家可能又要想:是不是每次返回的一定是try中的return语句呢?那么finally外的return不是一点作用没吗?请看下面

4、try块里的return语句在异常的情况下不会被执行,这样具体返回哪个看情况

  1. public class TestFinally {
  2. public static void main(String[] args) {
  3. System.out.println(test());
  4. }
  5. public static int test() {
  6. int b = 0;
  7. try {
  8. System.out.println("try block");
  9. b = b / 0;
  10. return b += 1;
  11. } catch (Exception e) {
  12. b += 10;
  13. System.out.println("catch block");
  14. } finally {
  15. b += 100;
  16. System.out.println("finally block");
  17. }
  18. return b;
  19. }
  20. }

运行结果是:

  1. try block
  2. catch block
  3. finally block
  4. 110

这里因 为在return之前发生了异常,所以try中的return不会被执行到,而是接着执行捕获异常的 catch 语句和最终的 finally 语句,此时两者对b的修改都影响了最终的返回值,这时最后的 return b 就起到作用了。

这里大家可能又有疑问:如果catch中有return语句呢?当然只有在异常的情况下才有可能会执行,那么是在 finally 之前就返回吗?看下面。

5、当发生异常后,catch中的return执行情况与未发生异常时try中return的执行情况完全一样

  1. public class TestFinally {
  2. public static void main(String[] args) {
  3. System.out.println(test());
  4. }
  5. public static int test() {
  6. int b = 0;
  7. try {
  8. System.out.println("try block");
  9. b = b / 0;
  10. return b += 1;
  11. } catch (Exception e) {
  12. b += 10;
  13. System.out.println("catch block");
  14. return 10086;
  15. } finally {
  16. b += 100;
  17. System.out.println("finally block");
  18. }
  19. //return b;
  20. }
  21. }

运行结果:

  1. try block
  2. catch block
  3. finally block
  4. 10086

说明了发生异常后,catch中的return语句先执行,确定了返回值后再去执行finally块,执行完了catch再返回,也就是说情况与try中的return语句执行完全一样。

总结:

  • finally块的语句在try或catch中的return语句执行之后返回之前执行
  • 且finally里的修改语句可能影响也可能不影响try或catch中return已经确定的返回值
  • 若finally里也有return语句则覆盖try或catch中的return语句直接返回

2019-3-4

异常 try catch finally return 执行关系 MD的更多相关文章

  1. finally return 执行关系 异常处理 c#

    Return.finally执行关系简述 除了函数出现system.exit(0)终止虚拟机,finally中的代码一定执行,return语句会等待finally的执行:如果是值传递,finally中 ...

  2. java try catch finally return执行

    public static int testBasic(){ int i = 1; try{ i++; System.out.println("try block, i = "+i ...

  3. try catch finally return之间的关系

    一.try catch finally return之间的关系: 正在写dsoFramer的时候,同事突然说面试的时候问的一个问题,catch和return那个先执行,我瞬间迷茫了,然后整理了整理,稍 ...

  4. 有return的情况下try catch finally的执行顺序(转)

    结论:1.不管有木有出现异常,finally块中代码都会执行:2.当try和catch中有return时,finally仍然会执行:3.finally是在return后面的表达式运算后执行的(此时并没 ...

  5. try catch finally的执行顺序(有return的情况下)

    结论:1.不管有木有出现异常,finally块中代码都会执行:2.当try和catch中有return时,finally仍然会执行:3.finally是在return后面的表达式运算后执行的(此时并没 ...

  6. 有return的情况下try catch finally的执行顺序

    结论:1.不管有木有出现异常,finally块中代码都会执行:2.当try和catch中有return时,finally仍然会执行:3.finally是在return后面的表达式运算后执行的(此时并没 ...

  7. 有return的情况下try catch finally的执行顺序(最有说服力的总结)

    结论:1.不管有木有出现异常,finally块中代码都会执行:2.当try和catch中有return时,finally仍然会执行:3.finally是在return后面的表达式运算后执行的(此时并没 ...

  8. Java基础知识强化之IO流笔记06:有return的情况下try catch finally的执行顺序

    1. 给出结论: (1)不管有木有出现异常,finally块中代码都会执行:(2)当try和catch中有return时,finally仍然会执行:(3)finally是在return后面的表达式运算 ...

  9. 【Java疑难杂症】有return的情况下try catch finally的执行顺序

    有这样一个问题,异常处理大家应该都不陌生,类似如下代码: public class Test { public static void main(String[] args) { int d1 = 0 ...

随机推荐

  1. JS定义一个立即执行的可重用函数

    我定义了一个函数表达式 testFun var testFun = (function() { ... //函数内容})(); 测试结果:虽然 testFun 函数有如愿在页面加载后立即被执行,但再次 ...

  2. linux命令: chown命令

    chown将指定文件的拥有者改为指定的用户或组,用户可以是用户名或者用户ID:组可以是组名或者组ID:文件是以空格分开的要改变权限的文件列表,支持通配符.系统管理员经常使用chown命令,在将文件拷贝 ...

  3. python 全栈开发,Day55(jQuery的位置信息,JS的事件流的概念(重点),事件对象,jQuery的事件绑定和解绑,事件委托(事件代理))

    一.jQuery的位置信息 jQuery的位置信息跟JS的client系列.offset系列.scroll系列封装好的一些简便api. 一.宽度和高度 获取宽度 .width() 描述:为匹配的元素集 ...

  4. 使用匿名内部类调用start方法

    package chapter03;//类实现接口public class WD implements Runnable{//重写接口的方法 @Override public void run() { ...

  5. Zbrush Topogun 备忘

    ====Zbrush==== 1.按住shift 在空白地方移动鼠标左键 就会去到正交视图 2.shift+F可以看一下布线的情况 3.按住shift 点一下画布,松开shift键,就可以旋转画布 4 ...

  6. canvas放射粒子效果

    这个也是别人的代码,就不多介绍了 写了些注释 body { overflow:hidden; margin:0; padding:0; background-color:#222222 } </ ...

  7. JS实现品字布局

    在网页后台中常用的布局是头部+侧边栏的形式 为了省去多于代码和重复修改多个页面 头部和侧边栏都是共用的,一直不改变的,所以写死在页面中. 中间的内容根据点击而发生改变,所以用iframe包起来 如何实 ...

  8. BZOJ1875 [SDOI2009]HH去散步 矩阵

    欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ1875 题意概括 在一个无向图(有重边无自环)中走,不能在经过连续经过某一条边2次. 现在走t步,问 ...

  9. 【Java】 剑指offer(50-2) 字符流中第一个只出现一次的字符

    本文参考自<剑指offer>一书,代码采用Java语言. 更多:<剑指Offer>Java实现合集   题目 请实现一个函数用来找出字符流中第一个只出现一次的字符.例如,当从字 ...

  10. 003 关于shell基础,大数据的前期准备

    一:正则表达式 1.通配符与正则表达式的区别 通配符有 *,?,[]. 区别: 通配符用来匹配文件名 正则表达式是为了匹配字符串 2.“.*”的意思 .:任意字符 *:匹配前一个字符任意次 3.过滤出 ...