共计 1716 个字符,预计需要花费 5 分钟才能阅读完成。
前言
Java 开发中,正常情况下,在 执行了 try 代码块 之后,finally中的代码一定会执行。我们实际开发也经常会利用这个特性,在 finally 中来执行一些特殊的操作,比如:释放资源、释放锁等。
demo
public class Finally {public static void main(String[] args) { | |
try { | |
// 正常业务逻辑 | |
System.out.println("I am try"); | |
throw new RuntimeException("I am RuntimeExecption"); | |
} catch (Exception e) { | |
// 异常处理 | |
System.out.println("I am Exception ->" + e.getMessage()); | |
} finally { | |
// 释放资源等 | |
System.out.println("I am Finally"); | |
} | |
} | |
} |
execute
I am try | |
I am Exception -> I am RuntimeExecption | |
I am Finally |
那么是不是 finally 中的代码一定会被执行呢?
其实不然,目前作者了解到有两种情况下,finally 中的代码不会被执行(不考虑 try 之前出现异常 或者 return 的情况,换言之,在 try 之前出现异常或者 return 时,try 对应的 finally 中的内容不会被执行)
finally 之前虚拟机被停止
demo
public class Finally {public static void main(String[] args) { | |
try { | |
// 正常业务逻辑 | |
System.out.println("I am try"); | |
throw new RuntimeException("I am RuntimeExecption"); | |
} catch (Exception e) { | |
// 异常处理 | |
System.out.println("I am Exception ->" + e.getMessage()); | |
System.exit(1);// 异常关闭虚拟机 | |
} finally { | |
// 释放资源等 | |
System.out.println("I am Finally"); | |
} | |
} | |
} |
execute
I am try | |
I am Exception -> I am RuntimeExecption |
上面的代码在出现了异常之后,使用 System.exit(1)退出关闭虚拟机,finally 中的代码当然无法执行。
守护线程中的 finally
demo
public class Finally {public static void main(String[] args) throws Exception {Thread thread = new Thread(new Runnable() {public void run() { | |
try { | |
// 正常业务逻辑 | |
System.out.println("I am try"); | |
Thread.sleep(1000); | |
} catch (Exception e) { | |
// 异常处理 | |
System.out.println("I am Exception ->" + e.getMessage()); | |
} finally { | |
// 释放资源等 | |
System.out.println("I am Finally"); | |
} | |
} | |
}); | |
thread.setDaemon(true); | |
thread.start(); | |
Thread.sleep(1000); | |
System.out.println("end"); | |
} | |
} |
execute
I am try | |
end |
使用 setDaemon(true) 方法来设置线程为守护线程,从打印结果中可以看到,守护线程中,try 代码块中的代码执行了,finally 代码块未必执行。主要原因是因为守护线程会随着所有非守护线程的退出而退出。上述在主线程和守护线程中都设置 sleep(1000) 的原因是怕线程 run() 方法还未开始执行主线程就退出了,这样的话 try 代码块中的内容都不会执行。当然不是说守护线程中的 finally 代码一定不会执行。
总结
java 中,如果想要执行 try 中的代码之后,不允许再执行 finally 中的代码,有以下两种方式:
- 使用 System.exit(1) 来退出虚拟机
- 把当前执行 trycatchfinally 代码的线程设置为守护线程
正文完
发表至: java
2019-04-30