java finally 代码块_finally代码块中的代码一定会执行吗?

熟悉java的人一定经常听说过,finally块中的代码一定会执行,但实际上真的是这样吗,本文带你看一下,java中有哪些情况finally中的代码不会执行。

先来解释下java中总是被人们放到一起比较的三个概念

final、finally、finalize的区别:

final: java中的修饰符。final修饰的类不能被继承,final修饰的方法不能被重写,final修饰的变量初始化之后不能被修改(当然这条不是绝对的,java中有一些手段可以修改)。

finally: java异常处理的组成部分,finally代码块中的代码一定会执行(如果真是这样本文也没必要存在了)。常用于释放资源。

finalize:Object类中的一个方法,垃圾收集器删除对象之前会调用这个对象的finalize方法。

finally与return:

public static int getNum(){

int a = 10;

try {

a = 20;

// a = a/0;

return a;

} catch (Exception e) {

a = 30;

return a;

}finally{

a = 40;

//return a;

}

}//调用该方法返回20

有不少人认为该方法的返回值是40。执行完finally中的代码后a的值是40这是毋庸置疑的,但方法执行的结果为什么是20呢?那是因为在执行finally之前,程序将return结果赋值给临时变量,然后执行finally代码块,最后将临时变量返回。当然如果在finally代码块中有return语句,最终生效的是finally代码块中的return。

如果不明白请看下面的例子。

public static void main(String[] args) {

System.out.println(getMap().get("name").toString());

//打印zhaoliu

}

public static Map getMap() {

Map map = new HashMap();

map.put("name", "zhangsan");

try {

map.put("name", "lisi");

return map;

}

catch (Exception e) {

map.put("name", "wangwu");

}

finally {

map.put("name", "zhaoliu");

map = null;

}

return map;

}

有人认为上述代码在main方法中会报错,事实并非如此。以上代码更加印证了return返回的不是变量本身,而是这里的"临时变量".这里也说明了java中只有值传递没有引用传递。如果不明的的可以看我的另一篇文章【为什么大家都说java是值传递】

finally中的代码真的一定会执行吗?

今天给大家列举四种finally中的代码不会执行的情况(如果还有其它情况可以在评论里告知,我们共同学习)。

1.在执行异常处理代码之前程序已经返回

public static boolean getTrue(boolean flag) {

if (flag) {

return flag;

}

try {

flag = true;

return flag;

} finally {

System.out.println("我是一定会执行的代码?");

}

}

如果上述代码传入的参数为true那finally中的代码就不会执行了。

2.在执行异常处理代码之前程序抛出异常

public static boolean getTrue(boolean flag) {

int i = 1/0;

try {

flag = true;

return flag;

} finally {

System.out.println("我是一定会执行的代码?");

}

}

这里会抛出异常,finally中的代码同样不会执行。原理同1中差不多,只有与 finally 相对应的 try 语句块得到执行的情况下,finally 语句块才会执行。

就算try语句执行了finally中的代码一定会执行吗,答案是no,请看下面两种情况。

3.finally之前执行了System.exit()

public static boolean getTrue(boolean flag) {

try {

flag = true;

System.exit(1);

return flag;

} finally {

System.out.println("我是一定会执行的代码?");

}

}

System.exit是用于结束当前正在运行中的java虚拟机,参数为0代表程序正常退出,非0代表程序非正常退出。道理也很简单整个程序都结束了,拿什么来执行finally呢。

4.所有后台线程终止时,后台线程会突然终止

public static void main(String[] args) {

Thread t1 = new Thread(new Runnable() {

@Override

public void run() {

try {

Thread.sleep(5);

} catch (Exception e) {

}finally{

System.out.println("我是一定会执行的代码?");

}

}

});

t1.setDaemon(true);//设置t1为后台线程

t1.start();

System.out.println("我是主线程中的代码,主线程是非后台线程。");

}

上述代码,后台线程t1中有finally块,但在执行前,主线程终止了,导致后台线程立即终止,故finally块无法执行(其实前面3点大家多多少少接触过,或是很容易理解,我主要想说的是这一点,一些人可能从来没听说过)。

总结:

1.与finally相对应的try语句得到执行的情况下,finally才有可能执行。

2.finally执行前,程序或线程终止,finally不会执行。

3.以后我再也不会说,fianlly代码块中的代码是一定会执行的代码了(哈哈)。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值