看几个例子,回顾一下执行顺序

例子1 无异常,finally中的return会导致提前返回

public static String test() {
   try {
       System.out.println("try");
       return "return in try";
  } catch(Exception e) {
       System.out.println("catch");
       return "return in catch";
  } finally {
       System.out.println("finally");
       return "return in finally";
  }
}

调用test()的结果:

try
finally
return in finally

例子2 无异常,try中的return会导致提前返回

public static String test() {
   try {
       System.out.println("try");
       return "return in try";
  } catch(Exception e) {
       System.out.println("catch");
  } finally {
       System.out.println("finally");
  }
   return "return in function";
}

调用test()的结果:

try
finally
return in try

例子3 有异常,finally中的return会导致提前返回

public static String test() {
       try {
           System.out.println("try");
           throw new Exception();
      } catch(Exception e) {
           System.out.println("catch");
           return "return in catch";
      } finally {
           System.out.println("finally");
           return "return in finally";
      }
  }

调用test()的结果:

try
catch
finally
return in finally

例子4 有异常,catch中的return会导致提前返回

public static String test() {
   try {
       System.out.println("try");
       throw new Exception();
  } catch(Exception e) {
       System.out.println("catch");
       return "return in catch";
  } finally {
       System.out.println("finally");
  }
}

调用test()的结果:

try
catch
finally
return in catch

例子4 有异常,不会提前返回

public static String test() {
   try {
       System.out.println("try");
       throw new Exception();
  } catch(Exception e) {
       System.out.println("catch");
  } finally {
       System.out.println("finally");
  }
   return "return in function";
}

调用test()的结果:

try
catch
finally
return in function

小结

上面这几个例子,大多数人已经非常了解。同时也衍生出一些理论,比如不要在finally中return等,不再赘述。

再看几个例子,返回值是否符合你的预期?

例子1

public static int test() {
   try {
       return 1;
  } finally {
       return 2;
  }
}

返回值:2

说明:与我们上面的例子一致,finally中的return导致提前返回,try中的return1不会被执行。

附编译后的代码:

public static int test() {
   try {
       boolean var0 = true;
       return 2;
  } finally {
      ;
  }
}

可以看到编译器做过优化,同时验证了boolean类型在底层是用int实现的,但注意你在源码中直接给int行赋值true或false是不被允许的。

例子2

public static int test() {
   int i;
   try {
       i = 3;
  } finally {
       i = 5;
  }
   return i;
}

返回值:5

说明:执行try中的代码后,再执行finally中的代码,最终i被赋值为5,最后返回

附编译后的代码:

public static int test() {
   boolean var0 = true;
   byte i;
   try {
       var0 = true;
  } finally {
       i = 5;
  }
   return i;
}

同样可以看出,编译器做了一些优化。

例子3

public static int test() {
   int i = 1;
   try {
       i = 3;
       return i;
  } finally {
       i = 5;
  }
}

返回值:3

这个例子稍微有点意思,按我们通常的思维,应该还是返回5,毕竟finally中把i赋值为5了嘛,然后由try中的return返回。然而很不幸,返回值是3。

为什么呢?先看一下编译后的代码:

public static int test() {
   boolean var0 = true;
   byte var1;
   try {
       int i = 3;
       var1 = i;
  } finally {
       var0 = true;
  }
   return var1;
}

我们会发现,finally中的代码块不起作用。不知你是否想起一点:Java中是按值传递的,finally中的i只是一个局部变量,finally块执行完毕后,局部变量便不复存在。

接着看例子:

例子4

public static List test() {
   List<Integer> list = new ArrayList<>();
   try {
       list.add(1);
       return list;
  } finally {
       list.add(2);
  }
}

返回:包含1和2两个元素的List对象。

说明:这个例子中,基本类型int被替换为引用类型List,虽然list是按值传递,但它内部的状态可变(体现在这里,就是可以add元素)。扩展:finally只能保证对象本身不可变,但无法保证对象内部状态不可变。

附编译后的代码:

public static List test() {
   ArrayList list = new ArrayList();
   ArrayList var1;
   try {
       list.add(1);
       var1 = list; // 执行这一步操作后,var1和list指向同一个对象
  } finally {
       list.add(2);
  }
   return var1;
}

你现在应该觉得自己理解了,那么再来看两个例子:

例子5

public static int test() {
    try {
        System.exit(0);
    } finally {
        return 2;
    }
}

该函数没有返回值。原因:jvm提前退出了。

附编译后的代码:

public static int test() {
   try {
       System.exit(0);
       return 2;
  } finally {
      ;
  }
}

例子6

public static int test() {
   try {
       while(true) {
          System.out.println("Infinite loop.");
      }
  } finally {
       return 2;
  }
}

由于try中的无限循环阻塞,永远执行不到finally中的代码块。

附编译后的代码:

public static int test() {
   try {
       while(true) {
           System.out.println("Infinite loop.");
      }
  } finally {
      ;
  }
}

小结

为了方便说明,只举了finally代码块的例子,catch代码块是类似的。

总结

执行顺序:

  1. try代码块中return前面的部分

  2. catch代码块中return前面的部分

  3. finally代码块中return前面的部分

  4. finally的return 或 catch的return 或 try的return。若前面的return被执行,会导致提前返回,同时后面的return被忽略。

  5. 方法的其他部分

变量:

  注意Java的按值传递规则

特殊情况:

  注意finally不会被执行的情况  

参考

Try-catch-finally-return clarification [duplicate]

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

你真的理解Java中的try/catch/finally吗?的更多相关文章

  1. Java内存管理-你真的理解Java中的数据类型吗(十)

    勿在流沙筑高台,出来混迟早要还的. 做一个积极的人 编码.改bug.提升自己 我有一个乐园,面向编程,春暖花开! 作为Java程序员,Java 的数据类型这个是一定要知道的! 但是不管是那种数据类型最 ...

  2. [转载]java中try 与catch的使用

    留着以后看 原文地址:与catch的使用">java中try 与catch的使用作者:碌碌如玉 try{ //代码区 }catch(Exception e){ //异常处理 } 代码区 ...

  3. 【Java学习笔记之三十三】详解Java中try,catch,finally的用法及分析

    这一篇我们将会介绍java中try,catch,finally的用法 以下先给出try,catch用法: try { //需要被检测的异常代码 } catch(Exception e) { //异常处 ...

  4. 深入理解Java中的不可变对象

    深入理解Java中的不可变对象 不可变对象想必大部分朋友都不陌生,大家在平时写代码的过程中100%会使用到不可变对象,比如最常见的String对象.包装器对象等,那么到底为何Java语言要这么设计,真 ...

  5. 深入理解Java中的IO

    深入理解Java中的IO 引言:     对程序语言的设计者来说,创建一个好的输入/输出(I/O)系统是一项艰难的任务 < Thinking in Java >   本文的目录视图如下: ...

  6. 理解Java中的ThreadLocal

    提到ThreadLocal,有些Android或者Java程序员可能有所陌生,可能会提出种种问题,它是做什么的,是不是和线程有关,怎么使用呢?等等问题,本文将总结一下我对ThreadLocal的理解和 ...

  7. JDK学习---深入理解java中的LinkedList

    本文参考资料: 1.<大话数据结构> 2.http://blog.csdn.net/jzhf2012/article/details/8540543 3.http://blog.csdn. ...

  8. 【Java】深入理解Java中的spi机制

    深入理解Java中的spi机制 SPI全名为Service Provider Interface是JDK内置的一种服务提供发现机制,是Java提供的一套用来被第三方实现或者扩展的API,它可以用来启用 ...

  9. 你真的理解Java 注解吗?

    你真的理解Java 注解吗? 1.什么是注解? 官方解释: Java 注解用于为 Java 代码提供元数据.作为元数据,注解不直接影响你的代码执行,但也有一些类型的注解实际上可以用于这一目的.Java ...

随机推荐

  1. 对称加密实现重要日志上报Openresty接口服务

    记录后端接收日志的流程: 由于记录的是广告数据,单次计费数据都会上报,全国内约10几万终端上报. 终端上报:Android电视端Apk上报 接收终端:Openresty(Nginx+lua)利用ngi ...

  2. Python将HTML转换为PDF

    Python将HTML转换为PDF 使用pdfkit库和wkhtmltopdf, pip install pdfkit wkhtmltopdflinux中一般需要添加sudo权限. Windows安装 ...

  3. django 发送邮件功能

    setting.py # 邮件配置 EMAIL_BACKEND = 'django.core.mail.backends.smtp.EmailBackend' EMAIL_HOST = 'smtp.e ...

  4. Python 获得程序 exe 的版本号

    Python 获得程序 exe 的版本号 python中需要安装 pywin32 包 # based on http://stackoverflow.com/questions/580924/pyth ...

  5. 在cubemx中使用freertos中的注意事项

    就是使用信号量等rtos自带特性的时候,务必先初始化然后在发生信号量或接收. 而且在中断中发送信号量或队列的时候,务必把使能中断的语句放在初始化freertos之后,尤其是cubemx生成的代码,默认 ...

  6. Unity 截图选择框,中间全透明,边缘半透明

    效果:点击白色框可拖拽选择区域 代码: using System.Collections; using System.Collections.Generic; using UnityEngine; u ...

  7. BZOJ 2527 [Poi2011]Meteors (整体二分+树状数组)

    整体二分板题,没啥好讲的-注意是个环-还有所有贡献会爆longlong,那么只要在加之前判断一下有没有达到需要的值就行了- CODE #include <set> #include < ...

  8. 安卓数据库sqllite查看工具Android Debug Database使用教程

    要使用Android Debug Database,首先得在安卓项目的build.gradle(module:app)文件中加入如下代码: debugCompile 'com.amitshekhar. ...

  9. 09 深科技相关表结构 (未完成)、git

    1.深科技相关 1. 深科技表结构(6表) 深科技4张+2张用户表 - 深科技 用户表 用户Token 文章来源 文章表 通用评论表 通用收藏表 # ######################## ...

  10. jsp前台输入框不输入值,后台怎么取出整型?

    当前台输入框限定整型,后台取值就会出现很多问题. eg: Integer.parseInt(request.getParameter("uno"));当前台的文本框不输入值,直接点 ...