前言

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 tryI am Exception -> I am RuntimeExecptionI 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 tryI 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 tryend

使用setDaemon(true)方法来设置线程为守护线程,从打印结果中可以看到,守护线程中,try代码块中的代码执行了,finally代码块未必执行。主要原因是因为守护线程会随着所有非守护线程的退出而退出。上述在主线程和守护线程中都设置sleep(1000)的原因是怕线程run()方法还未开始执行主线程就退出了,这样的话try代码块中的内容都不会执行。当然不是说守护线程中的finally代码一定不会执行。

总结

java中,如果想要执行try中的代码之后,不允许再执行finally中的代码,有以下两种方式:

  • 使用System.exit(1)来退出虚拟机
  • 把当前执行trycatchfinally代码的线程设置为守护线程