找工作时,有这样一道题:

try{}里面有一条return语句,那么紧跟在这个try后的finally{}里的代码会不会执行,什么时候执行,在return之前还是之后?

我没有怎么思考,根据脑子里仅有的一点异常知识,给出了下面的解:

如果try{}没有抛出异常,那么finally{}里的代码不会执行。

如果try{}抛出异常,那么finally{}里的代码会执行,在return之前。

借助下面的代码验证一下:

 private static int testFinally() {
try {
System.out.println("Try!");
return 1;
} finally {
System.out.println("Finally!");
}
}

毫无疑问,输出的结果显示上述解是草率的,是错的。那么,正解是什么?其实,读书用心点的话,可以注意到《Core Java》上的这段话:

Suppose you exit the middle of a try block with a return statement. Before the method returns, the finally block is executed. If the finally block also contains a return statement, then it masks the original return value.

——摘自《Core Java, Volume I: Fundamentals》

这段话就是正解。当方法中抛出异常,或包含break,continue,return语句时,方法就无法按照正常的方式终止。一般地,在try-finally结构中,当try部分意外终止时,并不意味着结束,方法仍然会继续执行finally部分。也就是说,上述代码返回1之前,会先打印“Finally!”。如果问原因,真说不出什么,没什么道理,Java就是这么干的。就和学英语遇到陌生的词组解释不了时,用“固定搭配”来安慰自己,是同一个道理。

进一步地,有没有某种情况,finally{}不会执行?另一个例子:

 private static void testFinally() {
try {
System.out.println("Try!");
System.exit(0);
} finally {
System.out.println("Finally!");
}
}

System.exit(0)可以阻止执行finally语句块。之所以可以做到,是因为System.exit()方法会立即终止当前正在运行的 Java 虚拟机。我们知道,在Java中,源码首先被编译成字节码,然后由虚拟机解释执行class文件。虚拟机进程被杀死后,程序当然无法接着执行下去。

众所周知,在需要关闭一些资源时,使用Finally语句是一种不错的选择。下面介绍一种结合try-catch和try-finally的方式:

 InputStream in = ...;
try{
try{
// [1]Code may throw excption.
}finally{
in.close();  //[2]
}
}catch(IOException e){
// [3]handle exception.
}

在这种方式中,内存的try语句块只有一个职责,就是关闭输入流。外层的try语句块也只有一个职责,就是捕获可能出现的异常。职责相当明确,结构相当清晰。但是,这种方式也存在潜在的错误:如果[2]处在关闭输入流时也抛出了IOException,那么[3]处捕获的异常究竟是[1]处抛出的,还是[2]处抛出的。事实上,若出现上述情况,则[3]处实际捕获到[2]处抛出的异常,会忽略[1]处抛出的异常。这是需要小心的地方!

以上是对异常,尤其是Finally语句易错点的总结,仅供参考。

由一道面试题想到的:Finally的更多相关文章

  1. 一道面试题:按照其描述要求用java语言实现快速排序

    回来想了想,写出了如下的程序: /** * 一道面试题,按照其描述要求进行快速排序(英文的,希望理解是对的..) * 要求:和一般的快速排序算法不同的是,它不是依次交换pivot和左右元素节点(交换2 ...

  2. 一道面试题与Java位操作 和 BitSet 库的使用

    前一段时间在网上看到这样一道面试题: 有个老的手机短信程序,由于当时的手机CPU,内存都很烂.所以这个短信程序只能记住256条短信,多了就删了. 每个短信有个唯一的ID,在0到255之间.当然用户可能 ...

  3. PHP递归创建多级目录(一道面试题的解题过程)

    今天看到一道面试题,要写出一个可以创建多级目录的函数: 我的第一个感觉就是用递归创建,具体思路如下: function Directory($dir){ if(is_dir($dir) || @mkd ...

  4. 关于Java类加载双亲委派机制的思考(附一道面试题)

    预定义类加载器和双亲委派机制 JVM预定义的三种类型类加载器: 启动(Bootstrap)类加载器:是用本地代码实现的类装入器,它负责将 <Java_Runtime_Home>/lib下面 ...

  5. 一道面试题比较synchronized和读写锁

    一.科普定义 这篇博文的两个主角“synchronized”和“读写锁” 1)synchronized 这个同步关键字相信大家都用得比较多,在上一篇“多个线程之间共享数据的方式”中也详细列举他的应用, ...

  6. <转>一道面试题比较synchronized和读写锁

    一.科普定义(原文:http://903497571.iteye.com/blog/1874752) 这篇博文的两个主角“synchronized”和“读写锁” 1)synchronized 这个同步 ...

  7. Java中有关构造函数的一道笔试题解析

    Java中有关构造函数的一道笔试题解析 1.详细题目例如以下 下列说法正确的有() A. class中的constructor不可省略 B. constructor必须与class同名,但方法不能与c ...

  8. 【C#基础知识】静态构造函数,来源于一道面试题的理解

    看到园友的一道面试题,很好奇,测试了一下结果. public class A { public static int X=B.Y ; public A() { ++X; } } public clas ...

  9. 一道笔试题来理顺Java中的值传递和引用传递

      题目如下: private static void change(StringBuffer str11, StringBuffer str12) { str12 = str11; str11 = ...

随机推荐

  1. effective c++ 条款12 copy all parts of an object

    这经常发生在更改代码的时候,当有自己的copy 赋值函数或者copy 构造函数时,编译器就不会维护这两个函数.导致发生遗忘. 可能出现的场景 class Customer { private: std ...

  2. POJ - 3249 Test for Job (DAG+topsort)

    Description Mr.Dog was fired by his company. In order to support his family, he must find a new job ...

  3. MongoDB学习笔记-基础概念

    mongodb中基本的概念 文档.集合.数据库 与关系数据库的概念对比更容易理解

  4. 玩转Web之Json(二)----jquery easy ui + Ajax +Json+SQL实现前后台数据交互

    最近在学Json,在网上也找过一些资料,觉得有点乱,在这里,我以easy ui的登录界面为例来说一下怎样用Json实现前后台的数据交互 使用Json,首先需要导入一些jar包,这些资源可以在网上下载到 ...

  5. Objective-C语言的一些基础特性

    OC与C++.Java等面向对象语言有很多的类似之处,不过在很多方面也是有所差别的.若是用过某一种面向对象语言,那么就很容易理解OC语言所用的范式和模板了.但是在语法使用上,也许会显得陌生.因为OC语 ...

  6. [cocos2dx注意事项014]一个用于cocos2dx对象智能指针模板

    现在,C++有许多实现智能指针,一个更无所谓.哈. 这种智能指针是专为cocos2dx 2.2.x自定义.主要的易用性,同时必须遵循现有的cocos2dx内存管理.特殊实现这样的智能指针.无需在使用时 ...

  7. oracle查看经常使用的系统信息

    总结了查看oracle数据库的经常使用sql ,有助于分析优化出一个健壮的系统程序来. 1.当前的数据库连接数: select count(*) from v$process 2.数据库同意的最大连接 ...

  8. Cocos2d-x 3.x plist+png 做动画

    ***************************************转载请注明出处:http://blog.csdn.net/lttree************************** ...

  9. Partitioner分区过程分析

    Partition中国人意味着分区,意义的碎片,这个阶段也是整个MapReduce该过程的第三阶段.在Map返回任务,是使key分到通过一定的分区算法.分到固定的区域中.给不同的Reduce做处理,达 ...

  10. zoj-3792-Romantic Value-最小割+数值转化

    假设不须要求边的个数的话,就是一个裸的最小割问题. 求边的个数就用边的权值记录一下. #include <stdio.h> #include <iostream> #inclu ...