Java程序员面试笔试宝典(第2版)
上QQ阅读APP看书,第一时间看更新

1.19 finally块中的代码什么时候被执行

问题描述:try{}里有一个return语句,那么紧跟在这个try后的finally{}里的code会不会被执行?什么时候被执行?在return前还是后?

在Java语言的异常处理中,finally语句块的作用就是保证无论出现什么情况,finally块里的代码一定会被执行。由于当程序执行return的时候就意味着结束对当前方法的调用并跳出这个方法体,任何语句要执行都只能在return前执行(除非碰到exit函数),因此finally块里的代码也是在return前执行的。此外,如果try-finally或者catch-finally中都有return,则finally块中的return语句将会覆盖别处的return语句,最终返回到调用者的是finally中return的值。下面通过一个例子(示例1)来说明这个问题:

程序运行结果为:

从上面这个例子中可以看出,在执行return前确实执行了finally中的代码。紧接着,在finally块里面放置return语句,例子(示例2)如下所示:

程序运行结果为:

从以上运行结果可以看出,当finally块中有return语句时,将会覆盖函数中其他return语句。此外,由于在一个方法内部定义的变量都存储在栈中,当这个函数结束后,其对应的栈就会被回收,此时在其方法体中定义的变量将不存在了,因此return在返回的时候不是直接返回变量的值,而是复制一份,然后返回。因此,对于基本类型的数据,在finally块中改变return的值对返回值没有任何影响,而对于引用类型的数据,就有影响。下面通过一个例子(示例3)来说明这个问题:

程序运行结果为:

程序在执行到return的时候会首先将返回值存储在一个指定的位置,然后去执行finally代码块,然后再返回。在方法testFinally1中调用return前首先把result的值1存储在一个指定的位置,然后再去执行finally块中的代码,此时修改result的值将不会影响到程序的返回结果。testFinally2中,在调用return前首先把s存储到一个指定的位置,由于s为引用类型,因此在finally块中修改s将会修改程序的返回结果。

引申:出现在Java程序中的finally代码块是不是一定会执行?

不一定会执行,下面给出两个finally代码块不会执行的例子。

1)当程序在进入try语句块之前就出现异常的时候,会直接结束,不会执行finally块中的代码。如下例所示:

程序运行结果为:

程序在执行int i=5/0的时候会抛出异常,导致没有执行try块,因此finally块也就不会被执行。

2)当程序在try块中强制退出的时候也不会去执行finally块中的代码,如下例所示:

程序运行结果为:

上例在try块中通过调用System.exit(0)强制退出了程序,因此导致finally块中的代码没有被执行。