在java中的finally關鍵一般與try一起使用,在程序進入try塊之後,無論程序是因為異常而中止或其它方式返回終止的,finally塊的內容一定會被執行,寫個例子來說明下:
public class TryAndFinallyTest { public static void main(String[] args) throws Exception{ try{ int a = testFinally(2); System.out.println("異常返回的結果a:"+a); }catch(Exception e ){ int b = testFinally(1); System.out.println("正常返回的結果b:"+b); } int b = testFinally(3); System.out.println("break返回的結果:" +b); b = testFinally(4); System.out.println("return返回的結果:"+b); } static int testFinally(int i) throws Exception{ int flag = i; try{//一旦進去try範圍無論程序是拋出異常或其它中斷情況,finally的內容都會被執行switch(i){ case 1:++i;break;//程序正常結束case 2:throw new Exception("測試下異常情況"); case 3:break; default :return -1; } }finally{ System.out.println("finally coming when i="+flag); } return i; } }
執行結果如下
finally coming when i=2finally coming when i=1正常返回的結果b:2finally coming when i=3break返回的結果:3finally coming when i=4return返回的結果:-1
結果說明無論上述什麼情況,finally塊總會被執行。
與其他語言的模型相比,finally 關鍵字是對Java 異常處理模型的最佳補充。 finally 結構使代碼總會執行,而不管有無異常發生。使用finally 可以維護對象的內部狀態,並可以清理非內存資源。 如果沒有finally,您的代碼就會很費解。例如,下面的代碼說明,在不使用finally 的情況下您必須如何編寫代碼來釋放非內存資源:
import java.net.*; import java.io.*; class WithoutFinally { public void foo() throws IOException { //在任一個空閒的端口上創建一個套接字ServerSocket ss = new ServerSocket(0); try { Socket socket = ss.accept(); //此處的其他代碼... } catch (IOException e) { ss.close(); //1 throw e; } //... ss.close(); / /2 } }
這段代碼創建了一個套接字,並調用accept 方法。在退出該方法之前,您必須關閉此套接字,以避免資源漏洞。為了完成這一任務,我們在//2 處調用close,它是該方法的最後一條語句。但是,如果try 塊中發生一個異常會怎麼樣呢?在這種情況下,//2 處的close 調用永遠不會發生。因此,您必須捕獲這個異常,並在重新發出這個異常之前在//1 處插入對close 的另一個調用。這樣就可以確保在退出該方法之前關閉套接字。
這樣編寫代碼既麻煩又易於出錯,但在沒有finally 的情況下這是必不可少的。不幸的是,在沒有finally 機制的語言中,程序員就可能忘記以這種方式組織他們的代碼,從而導致資源漏洞。 Java 中的finally 子句解決了這個問題。有了finally,前面的代碼就可以重寫為以下的形式:
import java.net.*; import java.io.*; class WithFinally { public void foo2() throws IOException { //在任一個空閒的端口上創建一個套接字ServerSocket ss = new ServerSocket(0); try { Socket socket = ss.accept(); //此處的其他代碼... } finally { ss.close(); } } }
finally 塊確保close 方法總被執行,而不管try 塊內是否發出異常。因此,可以確保在退出該方法之前總會調用close 方法。這樣您就可以確信套接字被關閉並且您沒有洩漏資源。在此方法中不需要再有一個catch 塊。在第一個示例中提供catch 塊只是為了關閉套接字,現在這是通過finally 關閉的。如果您確實提供了一個catch 塊,則finally 塊中的代碼在catch 塊完成以後執行。
finally 塊必須與try 或try/catch 塊配合使用。此外,不可能退出try 塊而不執行其finally 塊。如果finally 塊存在,則它總會執行。 (無論從那點看,這個陳述都是正確的。有一種方法可以退出try 塊而不執行finally 塊。如果代碼在try 內部執行一條System.exit(0); 語句,則應用程序終止而不會執行finally 執行。另一方面,如果您在try 塊執行期間撥掉電源,finally 也不會執行。)