1. 程式人生 > >有return情況下的try catch finally執行順序

有return情況下的try catch finally執行順序

結論:
1、不管有木有出現異常,finally塊中程式碼都會執行;
2、當try和catch中有return時,finally仍然會執行;
3、finally是在return後面的表示式運算後執行的(此時並沒有返回運算後的值,而是先把要返回的值儲存起來,管finally中的程式碼怎麼樣,返回的值都不會改變,任然是之前儲存的值),所以函式返回值是在finally執行前確定的;
4、finally中最好不要包含return,否則程式會提前退出,返回值不是try或catch中儲存的返回值。
舉例:

情況1:try{} catch(){}finally{} return;
            顯然程式按順序執行。
情況2

:try{ return; }catch(){} finally{} return;
          程式執行try塊中return之前(包括return語句中的表示式運算)程式碼;
         再執行finally塊,最後執行try中return;
         finally塊之後的語句return,因為程式在try中已經return所以不再執行。
情況3:try{ } catch(){return;} finally{} return;
         程式先執行try,如果遇到異常執行catch塊,
         有異常:則執行catch中return之前(包括return語句中的表示式運算)程式碼,再執行finally語句中全部程式碼,
                     最後執行catch塊中return. finally之後也就是4處的程式碼不再執行。
         無異常:執行完try再finally再return.
情況4
:try{ return; }catch(){} finally{return;}
          程式執行try塊中return之前(包括return語句中的表示式運算)程式碼;
          再執行finally塊,因為finally塊中有return所以提前退出。
情況5:try{} catch(){return;}finally{return;}
          程式執行catch塊中return之前(包括return語句中的表示式運算)程式碼;
          再執行finally塊,因為finally塊中有return所以提前退出。
情況6:try{ return;}catch(){return;} finally{return;}
          程式執行try塊中return之前(包括return語句中的表示式運算)程式碼;
          有異常:執行catch塊中return之前(包括return語句中的表示式運算)程式碼;
                       則再執行finally塊,因為finally塊中有return所以提前退出。
          無異常:則再執行finally塊,因為finally塊中有return所以提前退出。

最終結論
:任何執行try 或者catch中的return語句之前,都會先執行finally語句,如果finally存在的話。
                  如果finally中有return語句,那麼程式就return了,所以finally中的return是一定會被return的,
                  編譯器把finally中的return實現為一個warning。

下面是個測試程式
public class FinallyTest  
{
	public static void main(String[] args) {
		 
		System.out.println(new FinallyTest().test());;
	}

	static int test()
	{
		int x = 1;
		try
		{
			x++;
			return x;
		}
		finally
		{
			++x;
		}
	}
}
結果是2。

public class Test {

    public static void main(String[] args) {
        
        System.out.println(fun1());//2
    }

    @SuppressWarnings("finally")
    private static int fun1() {
        int i=0;
        try {
            i++;
            return i;
        }finally {
            ++i;
            return i;//return 在finally裡 提前返回
        }
    }    
}

 

public class Test {

    public static void main(String[] args) {
        
        System.out.println(fun1());//1
    }

    @SuppressWarnings("finally")
    private static int fun1() {
        int i=0;
        try {
            i++;
            return i;//先儲存i的值(1)
        }finally {
            ++i;//再執行finally i=2,但是並不影響 return的值 返回的依然是1
        }
    }    
}
分析:
	在try語句中,在執行return語句時,要返回的結果已經準備好了,就在此時,程式轉到finally執行了。
在轉去之前,try中先把要返回的結果存放到不同於x的區域性變數中去,執行完finally之後,在從中取出返回結果,
因此,即使finally中對變數x進行了改變,但是不會影響返回結果。
它應該使用棧儲存返回值。