(二十九)finally块中的代码什么时候执行

一、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);

 

 

 

 

 

 

posted @ 2019-02-19 19:29  测试开发分享站  阅读(112)  评论(0编辑  收藏  举报