java finally深入探究
When---什么时候需要finally:
在jdk1.7之前,所有涉及到I/O的相关操作,我们都会用到finally,以保证流在最后的正常关闭。jdk1.7之后,虽然所有实现Closable接口的流,可以通过在try块中定义,从而实现jvm自动关闭输入输出流。但其实在我们需要在代码块返回之前,实现在不管前面的操作是否执行成功,都要执行的某操作A。这时候我们就可以将A放入finally块中。很常见的一个操作就是锁的unlock操作。
What---什么是finally:
字面解释就是最终,eventually。其作用就是保证在try块中的代码执行完成之后,必然会执行finally中的语句。不管try块中是否抛出异常。
How---如何使用finally:
finally块必须配合try块使用,而不能单独使用。
Deeper---深入探究finally块:
在深入探究之前我们直接先给出四个结论:
1. finally块只会在try块执行的情况下才执行
2. finally块在离开try块执行完成后或者try块未执行完成但是接下来是控制转移语句时(return/continue/break)在控制转移语句之前执行。
3. 在执行finally语句之前,控制转移语句会将返回值存在本地变量中
4. finally块中的控制转移return语句能够覆盖try或者catch块中的返回值
其中1、2不做进一步说明。我们来对3和4直接通过两个例子进行深入理解:
关于3的例子:
1 public static void main(String[] args) {
2 System.out.println("return " + testFinally());
3 }
4
5 public static String testFinally() {
6 int i = 10;
7 try {
8 return i + "";
9 } finally {
10 i++; // 即使i的值改变,但是return语句涉及到的i的值已经保存在本地变量中
11 }
12 }
13
14 // 输出是10.
我们直接分析字节码得:
1 public static java.lang.String testFinally();
2 descriptor: ()Ljava/lang/String;
3 flags: ACC_PUBLIC, ACC_STATIC
4 Code:
5 stack=3, locals=3, args_size=0
6 0: bipush 10 --将10压入栈中
7 2: istore_0 --将栈顶元素(10)存储到局部变量0中
8 3: new #22 // class java/lang/StringBuilder
9 6: dup
10 7: iload_0 --从局部变量0中装载数据(10)入栈,后续代码是用来进行字符串拼接
11 8: invokestatic #47 // Method java/lang/String.valueOf:(I)Ljava/lang/String;
12 11: invokespecial #26 // Method java/lang/StringBuilder."<init>":(Ljava/lang/String;)V
13 14: invokevirtual #37 // Method java/lang/StringBuilder.toString:()Ljava/lang/String;
14 17: astore_2 --拼接后的结果存入局部变量2中(是10+"")
15 18: iinc 0, 1 --局部变量0的值加1
16 21: aload_2 --从局部变量2中装载数据(为10+"")
17 22: areturn --返回结果
18 23: astore_1 --将前面变量0加1的结果存入局部变量1
19 24: iinc 0, 1 --将局部变量0加1
20 27: aload_1 --装载局部变量1的值
21 28: athrow
其实areturn返回的是变量的引用值。这意味着,如果最终返回的是对象类型。那么在finally块执行之前,存储在本地变量中的只是对象引用的值,而不是对象的深拷贝。我们可以在finally块中对返回对象的属性进行改变。直接看下一个例子:
1 public static void main(String[] args) {
2 System.out.println("return " + testFinally());
3 }
4
5 public static Map<String, String> testFinally() {
6 Map<String, String> map = new HashMap<String, String>();
7 try {
8 map.put("aaa", "aaa");
9 return map;
10 } finally {
11 map.put("aaa", "bbb");
12 map = null;
13 }
14 }
15
16 // 输出为:return {aaa=bbb}
下面我们再来说说关于第4点的例子:
1 public static void main(String[] args) {
2 System.out.println("return " + testFinally());
3 }
4
5 private static String testFinally() {
6 try {
7 return "456";
8 } finally {
9 return "123";
10 }
11 }
12
13 // 输出:123
1 public static void main(String[] args) {
2 System.out.println("return " + testFinally());
3 }
4
5 private static String testFinally() {
6 try {
7 throw new SQLException();
8 //return "456";
9 } catch (SQLException e) {
10 System.out.println("catch SQLException!");
11 throw new RuntimeException("123");
12 }
13 finally {
14 return "123";
15 }
16 }
17
18 // 输出为:
19 catch SQLException!
20 return 123
更多的其他例子:
1 public static int getValue() {
2 int i = 1;
3 try {
4 i = 4;
5 } finally {
6 i++;
7 return i;
8 }
9 }
10 // 最终返回5
11
12 public static int getValue() {
13 int i = 1;
14 try {
15 i = 4;
16 } finally {
17 i++;
18 }
19
20 return i;
21 }
22 // 最终返回5
23
24 public static void main(String[] args) {
25 System.out.println(test());
26 }
27 public static String test() {
28 try {
29 System.out.println("try block");
30 return test1();
31 } finally {
32 System.out.println("finally block");
33 }
34 }
35 public static String test1() {
36 System.out.println("return statement");
37 return "after return";
38 }
39 // 最终输出:
40 try block
41 return statement
42 finally block
43 after return
参考链接:
https://www.ibm.com/developerworks/cn/java/j-lo-finally/
http://www.cnblogs.com/lanxuezaipiao/p/3440471.html
http://blog.csdn.net/michealmc/article/details/52237639
- C++拷贝构造函数(深拷贝,浅拷贝)
- 安装glog和gflags
- FFmpeg_3.2.4+SDL_2.0.5学习(1)音视频解码帧及显示/播放数据
- FFmpeg_3.2.4+SDL_2.0.5学习(2)视频同步基础
- ubuntu17.04更换主题
- ubuntu17.04新安装之后的软件准备
- 打造一流编辑器vimplus
- ffmpeg+sdl播放类
- 拉丁猪文字游戏
- 在Windows Mobile的控制台应用中使用Notification
- vim使用经验积累
- Spring+Mybatis+Maven+Mysql编程实战
- LD_LIBRARY_PATH和LIBRARY_PATH的区别
- c++11的Condition_variable
- java教程
- Java快速入门
- Java 开发环境配置
- Java基本语法
- Java 对象和类
- Java 基本数据类型
- Java 变量类型
- Java 修饰符
- Java 运算符
- Java 循环结构
- Java 分支结构
- Java Number类
- Java Character类
- Java String类
- Java StringBuffer和StringBuilder类
- Java 数组
- Java 日期时间
- Java 正则表达式
- Java 方法
- Java 流(Stream)、文件(File)和IO
- Java 异常处理
- Java 继承
- Java 重写(Override)与重载(Overload)
- Java 多态
- Java 抽象类
- Java 封装
- Java 接口
- Java 包(package)
- Java 数据结构
- Java 集合框架
- Java 泛型
- Java 序列化
- Java 网络编程
- Java 发送邮件
- Java 多线程编程
- Java Applet基础
- Java 文档注释
- 前端成神之路-CSS初识
- 前端成神之路-CSS基础选择器
- 前端成神之路-CSS文字文本样式
- 前端成神之路-CSS(选择器、背景、特性)
- 前端成神之路-列表和表单
- 前端成神之路-HTML
- Windows下如何查看某个端口被占用,以及如何杀死某个进程
- VSCode插件Todo Tree如何忽略某些文件夹?
- Express新建项目与配置项目热加载
- selenium控制已打开的页面
- 盘点一下lua脚本和python的区别(基础)
- python识别视频黑屏或者低清晰度
- python在linux环境下安装skimage
- python3异步爬虫 ——aiohttp模板使用
- [PHP] 抽象工厂设计模式-创建型设计模式