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的例子:

 public static void main(String[] args) {
System.out.println("return " + testFinally());
} public static String testFinally() {
int i = 10;
try {
return i + "";
} finally {
i++; // 即使i的值改变,但是return语句涉及到的i的值已经保存在本地变量中
}
} // 输出是10.

我们直接分析字节码得:

  public static java.lang.String testFinally();
descriptor: ()Ljava/lang/String;
flags: ACC_PUBLIC, ACC_STATIC
Code:
stack=3, locals=3, args_size=0
0: bipush 10           --将10压入栈中
2: istore_0                --将栈顶元素(10)存储到局部变量0中
3: new #22 // class java/lang/StringBuilder
6: dup
7: iload_0                --从局部变量0中装载数据(10)入栈,后续代码是用来进行字符串拼接
8: invokestatic #47 // Method java/lang/String.valueOf:(I)Ljava/lang/String;
11: invokespecial #26 // Method java/lang/StringBuilder."<init>":(Ljava/lang/String;)V
14: invokevirtual #37 // Method java/lang/StringBuilder.toString:()Ljava/lang/String;
17: astore_2               --拼接后的结果存入局部变量2中(是10+"")
18: iinc 0, 1         --局部变量0的值加1
21: aload_2                --从局部变量2中装载数据(为10+"")
22: areturn                --返回结果
23: astore_1               --将前面变量0加1的结果存入局部变量1
24: iinc 0, 1         --将局部变量0加1
27: aload_1                --装载局部变量1的值
28: athrow

其实areturn返回的是变量的引用值。这意味着,如果最终返回的是对象类型。那么在finally块执行之前,存储在本地变量中的只是对象引用的值,而不是对象的深拷贝。我们可以在finally块中对返回对象的属性进行改变。直接看下一个例子:

 public static void main(String[] args) {
System.out.println("return " + testFinally());
} public static Map<String, String> testFinally() {
Map<String, String> map = new HashMap<String, String>();
try {
map.put("aaa", "aaa");
return map;
} finally {
map.put("aaa", "bbb");
map = null;
}
} // 输出为:return {aaa=bbb}

下面我们再来说说关于第4点的例子:

 public static void main(String[] args) {
System.out.println("return " + testFinally());
} private static String testFinally() {
try {
return "456";
} finally {
return "123";
}
} // 输出:123
 public static void main(String[] args) {
System.out.println("return " + testFinally());
} private static String testFinally() {
try {
throw new SQLException();
//return "456";
} catch (SQLException e) {
System.out.println("catch SQLException!");
throw new RuntimeException("123");
}
finally {
return "123";
}
} // 输出为:
catch SQLException!
return 123

更多的其他例子:

 public static int getValue() {
int i = 1;
try {
i = 4;
} finally {
i++;
return i;
}
}
// 最终返回5 public static int getValue() {
int i = 1;
try {
i = 4;
} finally {
i++;
} return i;
}
// 最终返回5 public static void main(String[] args) {
System.out.println(test());
}
public static String test() {
try {
System.out.println("try block");
return test1();
} finally {
System.out.println("finally block");
}
}
public static String test1() {
System.out.println("return statement");
return "after return";
}
// 最终输出:
try block
return statement
finally block
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

java finally深入探究的更多相关文章

  1. (转)Java动态追踪技术探究

    背景:美团的技术沙龙分享的文章都还是很不错的,通俗易懂,开阔视野,后面又机会要好好实践一番. Java动态追踪技术探究 楔子 jsp的修改 重新加载不需要重启servlet.如何在不重启jvm的情况下 ...

  2. Java动态追踪技术探究(动态修改)

    Java动态追踪技术探究 Java探针-Java Agent技术-阿里面试题 秒懂Java动态编程(Javassist研究) 可以用于在类加载的时候,修改字节码. Java agent(Java探针) ...

  3. Java NIO的探究

    1.Java NIO与阻塞IO的区别 阻塞IO通信模型(在上一篇<J2SE网络编程之 TCP与UDP>博客中有所介绍) 我们知道阻塞I/O在调用InputStream.read()方法时是 ...

  4. Java 8特性探究(1):通往lambda之路与 lambda表达式10个示例

    本文由 ImportNew 函数式接口 函数式接口(functional interface 也叫功能性接口,其实是同一个东西).简单来说,函数式接口是只包含一个方法的接口.比如Java标准库中的ja ...

  5. java中事件处理探究

    事件的触发可以源于用户,也可以用代码来主动设置事件的发生.如setSelected()java.awt.event中 听众接口 事件类 适配器类 ComponentListener     Conta ...

  6. Java动态追踪技术探究

    引子 在遥远的希艾斯星球爪哇国塞沃城中,两名年轻的程序员正在为一件事情苦恼,程序出问题了,一时看不出问题出在哪里,于是有了以下对话: “Debug一下吧.” “线上机器,没开Debug端口.” “看日 ...

  7. 【基本功】Java动态追踪技术探究 不重启JVM,替换掉已经加载的类?不重启JVM,获知运行时对象的属性

    https://mp.weixin.qq.com/s/_hSaI5yMvPTWxvFgl-UItA 小结: 1.根据Java的类加载机制,在同一个ClassLoader中,类是不允许重复的: 2.单例 ...

  8. 对Java字符串的探究

    问题的出发点 在网上看到一道题: 1 String str = new String("abc"); 以上代码执行过程中生成了多少个 String 对象? 答案写的是两个.&quo ...

  9. Java网络编程探究|乐字节

    大家好,我是乐字节小乐,上次给大家讲述了Java中的IO流之输出流|乐字节,本文将会给大家讲述网络编程. 主要内容如下: 网络 网络分层 IP位置 端口port 网络编程 一. 网络 1.概念 网络即 ...

随机推荐

  1. a:hover标签已经定义了text-decoration:none,并且生效,但是还是有下划线

    a标签在F12计算出来的样式里 text-decoration:none; 确实有被应用到.但是链接的下划线并没有被去掉... 解决办法:p:commandLink <p:commandLink ...

  2. Mac_OS_Sierra_10.12.6编译OpenJDK9

    编译环境以及依赖 macOS:Sierra,10.12.6 处理器:2.6 GHz Intel Core i7 内存:16 GB 2133 MHz LPDDR3 Command Line Tools ...

  3. py2 to py3 return iterator

    Views And Iterators Instead Of Lists Some well-known APIs no longer return lists: dict methods dict. ...

  4. python实现二分查找算法

    二分查找算法也成为折半算法,对数搜索算法,一会中在有序数组中查找特定一个元素的搜索算法.搜索过程是从数组中间元素开始的 如果中间元素正好是要查找的元素,则搜索过程结束:如果查找的数大于中间数,则在数组 ...

  5. 启用composer镜像服务

    使用composer下载东西,需要FQ时,可使用其镜像服务 安装composer后,命令行执行全局配置 composer config -g repo.packagist composer https ...

  6. 微信小程序 - 上拉加载

    demo.wxml  文件 <view wx:for="{{listdata}}" wx:key="listdata" class='listitem'& ...

  7. 【LCT】BZOJ3091 城市旅行

    3091: 城市旅行 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1927  Solved: 631[Submit][Status][Discuss ...

  8. 【转载】漫谈HADOOP HDFS BALANCER

    Hadoop的HDFS集群非常容易出现机器与机器之间磁盘利用率不平衡的情况,比如集群中添加新的数据节点.当HDFS出现不平衡状况的时候,将引发很多问题,比如MR程序无法很好地利用本地计算的优势,机器之 ...

  9. css的学习笔记

    CSS3有哪些新特性? 1. CSS3实现圆角(border-radius),阴影(box-shadow), 2. 对文字加特效(text-shadow.),线性渐变(gradient),旋转(tra ...

  10. 激活IDEA

    1 先跳过提示 2 点击regist 3 输入注册码  : 我的注册码在这里找的 - >>  http://hw1287789687.iteye.com/blog/2153894