面试中关于Java异常处理finally是否执的总结归纳
首先我们区分一下这三个关键词:
final: java中的修饰符。final修饰的类不能被继承,final修饰的方法不能被重写,final修饰的变量初始化之后不能被修改(当然这条不是绝对的,java中有一些手段可以修改)。
finally: java异常处理的组成部分,finally代码块中的代码一定会执行(如果真是这样本文也没必要存在了)。常用于释放资源。
try{
}catch(异常 a){
}finally{
}
finalize:Object类中的一个方法,垃圾收集器删除对象之前会调用这个对象的finalize方法。
/------------------------------------------------------------------------------------------------------------------------/
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语句。那么try语句执行了finally中的代码一定会执行吗?我们看下面两个栗子
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块无法执行)。
总结:
1.虽然finally在Java中是必然执行的,这个必然执行仅限于与finally相对应的try语句得到执行的情况下,finally才有可能执行。
2.finally执行前,程序或线程执行者(电脑、控制台等等)断电或者被关闭终止进程,finally不再执行。