一、finally代码在return前执行
①finally代码一定会执行
②return表示退出函数体,返回函数调用处
综合①②finally代码会在return前执行
即程序在执行到return时会首先将返回值存储在一个指定的位置,其次去执行finally块,再返回。
public static int testFinally()
{
try{
return 1;
}catch(Exception e)
{
return 0;
}finally{
System.out.println("execute finally");
}
}
//execute finally
1
二、如果try-finally或者catch-finally中都有return,那么finally块中的return语句将会覆盖别处的return语句,最终但会到调用者哪里的是finally中的return的值。
public static int testFinally()
{
try{
return 1;
}catch(Exception e)
{
return 0;
}finally{
System.out.println("execute finally");
return 3;
}
}
//执行结果:
execute finally
3
三、return基本数据类型和引用类型
对于基本数据类型,在finally块中改变return的值对返回值没有任何影响,而对引用类型的数据会有影响。
(由于在一个方法内部定义的变量都存储在栈中,当这个函数结束后,其对应的栈就会被回收,此时在其方法体中定义的变量就不存在了,因此return在返回时不是直接返回变量的值,而是复制一份,然后返回)
即
public static int testFinally()
{
int result=1;
try{
result=2;
return result;
}catch(Exception e)
{
return 0;
}finally{
result=3;
System.out.println("execute finally");
//return 4; 若这里写return,则返回的结果为4(因为finally中的return会覆盖其他部分的return)
//若不写return,则返回的结果为2,(因为finally执行完,再到try中return )
}
}
在方法testFinally1中调用return前,先将result的值1存储在一个指定位置,然后再去执行finally块中的代码,此时修改result的值将不会影响到程序的返回结果。
public static StringBuffer testFinally()
{
StringBuffer s=new StringBuffer("hello");
try{
return s;
}catch(Exception e)
{
return null;
}finally{
s.append(" world");
System.out.println("execute finally");
}
}
//输出:
execute finally
hello world
testFinally2中,在调用return前首先把s存储到一个指定的位置,由于s为引用类型,s存储的是对象的地址,因此在finally块中修改s将会修改程序的返回结果。
四、在java程序中的finally块不一定会被执行
①在进入try之前就出异常
public static void testFinally()
{
int i=5/0;
try{
System.out.println("try block");
}catch(Exception e)
{
System.out.println("try block");
}finally{
System.out.println("try block");
}
}
//会报java.lang.ArithmeticException: / by zero 异常,之后代码不会执行
②try块中强制退出System.exit(0);