finally块的作用
通常用于处理善后工作。当try块里出现异常时,会立即跳出try块,到catch块匹配对应的异常,执行catch块里的语句。此时,可能在try块里存在打开的文件没关闭,连接的网络没断开,这部分资源是GC所不能自动处理的,所以finally的作用就是将它们及时释放回收。
finally块不被执行的情况,总共有3种:不进入try块、程序中止、线程中止(带finally块的是守护线程,其非守护线程都执行完毕)。
public class FinallyTest {
//finally不执行情况
public static void main(String[] args) {
//1.不进入try块
boolean flag = false;
if (flag) {
try {
System.out.println("flag");
} finally {
System.out.println("finally");
}
}
//2.程序结束。相当于宕机
try {
System.out.println("flag");
System.exit(0);
} finally {
System.out.println("finally");
}
//3.无线循环
try {
while (true){
System.out.println("flag");
}
} finally {
System.out.println("finally");
}
//4.非守护线程终止
Thread thread = new Thread(new Task());
thread.setDaemon(true);
thread.start();
TimeUnit.SECONDS.sleep(1);
System.out.println("test");
System.out.println("test1");
/*
4.上述代码,语句 TimeUnit.SECONDS.sleep(5); 会使main线程阻塞5秒,足够线程thread执行。
如果将该语句注释,非守护线程main线程执行完 thread.start();
这行后,存在三种情况:①CPU时间片还是交给main线程,则非守护线程执行完毕,守护线程thread就会被终止,finally块不执行;
②CPU时间片交给thread线程,但是thread线程刚执行完try块,就得交付时间片给main,main已经无语句执行,就会结束,
导致守护线程thread也要结束,finally块不执行;③CPU时间片交付thread线程,thread线程完全执行,finally块被执行。
*
*/
}
}
其他迷惑行为
当try块里面包含有break,return,continue,该次try块结束后,finally块也会执行。
//当try块里面包含有break,continue,return,该次try块结束后,finally块也会执行。
for (int i = 0; i < 5; i++) {
try {
if (i == 2) {
System.out.println(i);
// break;
// return;
continue;
}
} finally {
System.out.println(i);
}
}