finally块中的代码什么时候被执行?finally是不是一定会被执行?(见最后引申)
问题描述:try{}里有一个return语句,那么紧跟在这个try{}后面的finally{}中的代码是否会被执行?如果会的话,什么时候被执行,在return之前还是return之后?
在Java语言的异常处理中,finally块的作用就是为了保证无论出现什么情况,finally块里的代码一定会被执行。由于程序执行return就意味着结束对当前函数的调用并跳出这个函数体,因此任何语句要执行都只能在return前执行(除非碰到exit函数),因此finally块里的代码也是在return之前执行的。此外,如果try-finally或者catch-finally中都有return,那么finally块中的return将会覆盖别处的return语句,最终返回到调用者那里的是finally中return的值。下面通过一个例子来说明这个问题:
- package com.js;
- /**
- * try-catch中有return语句,finally中代码运行时机问题
- * @author jiangshuai
- */
- public class Test{
- public static int testFinally(){
- try {
- return 1;
- } catch (Exception e) {
- return 0;
- }finally{
- System.out.println("execute finally");
- }
- }
- public static void main(String[] args){
- int result = testFinally();
- System.out.println(result);
- }
- }
运行结果:
execute finally
1
从上面这个例子中可以看出,在执行return语句前确实执行了finally块中的代码。紧接着,在finally块里放置个return语句,来看看到底最终返回的是哪个return语句的值,例子如下图所示:
- package com.js;
- /**
- * try-catch中有多个return语句,研究return的是哪一个
- * @author jiangshuai
- */
- public class Test{
- public static int testFinally(){
- try {
- return 1;
- } catch (Exception e) {
- return 0;
- }finally{
- System.out.println("execute finally");
- return 3;
- }
- }
- public static void main(String[] args){
- int result = testFinally();
- System.out.println(result);
- }
- }
运行结果:
execute finally
3
从以上运行结果可以看出,当finally块中有return语句时,将会覆盖函数中其他return语句。此外,由于在一个方法内部定义的变量都存储在栈中,当这个函数结束后,其对应的栈就会被回收,此时在其方法体中定义的变量将不存在了,因此,对基本类型的数据,在finally块中改变return的值对返回值没有任何影响,而对引用类型的数据会有影响(详见
Java中的值传递与引用传递详解
)。下面通过一个例子来说明这个问题:
- package com.js;
- /**
- * 在finally块中改变基本数据类型、引用类型对比
- * @author jiangshuai
- */
- public class Test{
- public static int testFinally1(){
- int result = 1;
- try {
- result = 2;
- return result;
- } catch (Exception e) {
- return 0;
- }finally{
- result = 3;
- System.out.println("execute finally1");
- }
- }
- public static StringBuffer testFinally2(){
- StringBuffer s = new StringBuffer("Hello");
- try {
- return s;
- } catch (Exception e) {
- return null;
- }finally{
- s.append(" World");
- System.out.println("execute finally2");
- }
- }
- public static void main(String[] args){
- int result = testFinally1();
- System.out.println(result);
- StringBuffer resultRef = testFinally2();
- System.out.println(resultRef);
- }
- }
运行结果:
execute finally1
2
execute finally2
Hello World
程序在执行到return时会首先将返回值存储在一个指定的位置,其次去执行finally块,最后再返回。在方法testFinally1中调用return前,先把result的值1存储在一个指定的位置,然后再去执行finally块中的代码,此时修改result的值将不会影响到程序的返回结果。testFinally2中,在调用return前先把s存储到一个指定的位置,由于s为引用类型,因此在finally中修改s将会修改程序的返回结果。
引申:出现在Java程序中的finally块是不是一定会被执行?
答案:不一定。
下面给出两个finally块不会被执行的例子:
1)、当程序进入try块之前就出现异常时,会直接结束,不会执行finally块中的代码,示例如下:
- package com.js;
- /**
- * 在try之前发生异常
- * @author jiangshuai
- */
- public class Test{
- public static void testFinally1(){
- int result = 1/0;
- try {
- System.out.println("try block");
- } catch (Exception e) {
- System.out.println("catch block");
- }finally{
- System.out.println("finally block");
- }
- }
- public static void main(String[] args){
- testFinally1();
- }
- }
运行结果:
Exception in thread "main" java.lang.ArithmeticException: / by zero
at com.js.Test.testFinally1(Test.java:9)
at com.js.Test.main(Test.java:19)
程序在执行1/0时会抛出异常,导致没有执行try块,因此finally块也就不会被执行。
2)、当程序在try块中强制退出时也不会去执行finally块中的代码,示例如下:
- package com.js;
- /**
- * 在try之前发生异常
- * @author jiangshuai
- */
- public class Test{
- public static void testFinally1(){
- try {
- System.out.println("try block");
- System.exit(0);
- } catch (Exception e) {
- System.out.println("catch block");
- }finally{
- System.out.println("finally block");
- }
- }
- public static void main(String[] args){
- testFinally1();
- }
- }
运行结果:
try block
上例在try块中通过调用System.exit(0)强制退出了程序,因此导致finally块中的代码没有被执行。
finally块中的代码什么时候被执行?finally是不是一定会被执行?(见最后引申)相关推荐
- finally块中的代码一定会执行吗?
为什么80%的码农都做不了架构师?>>> 在Sun Tutorial中有这样一句话:The finally block always executes when the try ...
- junit mockito_从工作中清除代码–使用JUnit 5,Mockito和AssertJ编写可执行规范
junit mockito 可执行规范是也可以用作设计规范的测试. 通过启用通用语言(在DDD世界中,这也称为无处不在的语言 ),它们使技术和业务团队能够进入同一页面. 它们充当代码的未来维护者的文档 ...
- 从工作中清除代码–使用JUnit 5,Mockito和AssertJ编写可执行规范
可执行规范是可以用作设计规范的测试. 通过启用公共语言(在DDD世界中,这也称为无处不在的语言 ),它们使技术和业务团队能够进入同一页面. 它们充当代码的未来维护者的文档. 在本文中,我们将看到一种编 ...
- Java 中静态代码块 static的作用及用法
java静态代码块的作用: Java静态代码块中的代码会在类加载JVM时运行,且只被执行一次,也就是说这些代码不需要实例化类就能够被调用.一般情况下,如果有些代码必须在项目启动的时候就执行的时候,就需 ...
- {转载}java Finally块中代码什么时候执行
转载自:https://blog.csdn.net/Dove_Knowledge/article/details/71077512 原创作者:Yes_JiangShuai 问题描述:try{}里有一个 ...
- 阿里面试题,为什么wait()方法要放在同步块中?
某天我在***的时候,突然有个小伙伴微信上说:"哥,阿里面试又又挂了,被问到为什么wait()方法要放在同步块中,没答出来!" 我顿时觉得**一紧,仔细回顾一下,如果wait()方 ...
- java同步方法同步块_java 同步代码块与同步方法
同步代码块 synchronized(obj) {//代码块 } obj 为同步监视器,以上代码的含义为:线程开始执行同步代码块(中的代码)之前,必须先获得对同步监视器的锁定. 代码块中的代码是执行代 ...
- java fianlly_Java冷知识:finally中的代码一定会执行吗?
前言 Java开发中,正常情况下,在执行了try代码块之后,finally中的代码一定会执行.我们实际开发也经常会利用这个特性,在finally中来执行一些特殊的操作,比如:释放资源.释放锁等. de ...
- Static静态代码块以及各代码块之间的执行顺序
前言 在Java中,Static静态代码块.构造代码块.普通代码块.构造函数的执行顺序是一个java学习者必须掌握的基础,本篇博客旨在让大家能清除了解它们之间的执行顺序. 文章目录 代码块的分类 1. ...
最新文章
- Action4D:人群和杂物中的在线动作识别:CVPR209论文阅读
- Ubuntu上通过android toolchain交叉编译Valgrind操作步骤
- requests 两种传参方式
- gsl eclipse
- Django视图(一)
- go 异常捕获处理 panic defer recover
- Linux并发服务器编程之多线程并发服务器
- 【高并发】面试官:讲讲高并发场景下如何优化加锁方式?
- 学术人必备!懒人制作学术会议Oral/Spotlight Video指南
- 【LeetCode】剑指 Offer 51. 数组中的逆序对
- 美团股价盘中涨幅超7% 市值重回2万亿港元关口
- ISA频繁断线的原因和解决方法
- 阿里发布的规约插件,细节都在这里
- 如何从JavaScript对象中删除键? [重复]
- python 标准库 excel_Python 操作 Excel 的函数库
- WebRTC入门知识
- 今天突然出现了Property IsLocked is not available for Login '[sa]',我太阳,下面有绝招对付它!...
- 开源crm客户关系统管理系统源码,免费分享
- php高德根据ip获取经纬度,开放平台:高德地图获取经纬度
- 游戏视频录制用什么软件?这2款软件,用过都说好