不知道大家有没有问过自己,finally方法会在什么时候不执行呢?那我们下面来看看finally到底会不会不执行。
1.finally的含义
finally的真正含义是指从try代码块出来才一定会执行相应的finally代码块。
代码语言:javascript复制public class Test {
public static void main(String[] args) {
foo(false);
}
public static void foo(boolean flag) {
System.out.println("enter foo()");
if(flag) {
try {
System.out.println("enter try block");
} finally {
System.out.println("enter finally block");
}
} else {
System.out.println("leave foo()");
}
}
}
/******************
控制台打印如下
enter foo()
leave foo()
*******************/
上述代码,flag为false,没有进入try代码块,对应的finally自然也不会执行。
2.System.exit()
System.exit()的作用是中止当前虚拟机,虚拟机都被中止了,finally代码块自然不会执行。
代码语言:javascript复制public class Test {
public static void main(String[] args) {
foo();
}
public static void foo() {
System.out.println("enter foo()");
try {
System.out.println("enter try block");
System.exit();
} finally {
System.out.println("enter finally block");
}
}
}
/*****************
控制台打印如下
enter foo()
enter try block
******************/
上述代码,进入foo()方法后再进入try代码块,但是在进入finally代码块之前调用了System.exit()中止虚拟机, finally代码块不会被执行。
3.守护(daemon)线程被中止时
java线程分为两类,守护线程和非守护线程。当所有的非守护线程中止时,不论存不存在守护线程,虚拟机都会kill掉守护线程从而中止程序。 虚拟机中,执行main方法的线程就是一个非守护线程,垃圾回收则是另一个守护线程,main执行完,程序就中止了,而不管垃圾回收线程是否中止。 所以,如果守护线程中存在finally代码块,那么当所有的非守护线程中止时,守护线程被kill掉,其finally代码块是不会执行的。
代码语言:javascript复制public class Test {
public static void main(String[] args) {
Thread t = new Thread(new Task());
t.setDaemon(true); //置为守护线程
t.start();
try {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
throw new RuntimeException("the " Thread.currentThread().getName() " has been interrupted",e);
}
}
}
class Task implements Runnable {
@Override
public void run() {
System.out.println("enter run()");
try {
System.out.println("enter try block");
TimeUnit.SECONDS.sleep(5); //阻塞5s
} catch(InterruptedException e) {
System.out.println("enter catch block");
throw new RuntimeException("the " Thread.currentThread().getName() " has been interrupted",e);
} finally {
System.out.println("enter finally block");
}
}
}
/*******************
控制台打印如下
enter run()
enter try block
********************/
上述代码,main()执行完,非守护线程也就结束了,虽然线程t处于阻塞状态,但由于其是守护线程,所以程序仍会中止。 而且,即使其进入了try代码块,finally代码块也不会被执行。
总结
finally代码块并非一定执行,在不进入try代码块或者程序被中止时就不会执行。所以写代码一定不要想当然,可能你认为绝对不会执行的代码,有时候它还真会执行,什么时候我们都要带着批判思维(熟话说:带脑子)进行程序的开发。