淺談Java中final,finalized,finally

歡迎訪問本人博客!http://blog.csdn.net/ktb2007

final:

final可以讓你控制你的成員、方法或者是一個類是否可被覆寫或繼承等功能,這些特點使final在Java中擁有了一個不可或缺的地位,也是學習Java時必須要知道和掌握的關鍵字之一。

final成員

當你在類中定義變量時,在其前面加上final關鍵字,那便是說,這個變量一旦被初始化便不可改變,這裏不可改變的意思對基本類型來說是其值不可變,而對於對象變量來說其引用不可再變。其初始化可以在兩個地方,一是其定義處,二是在構造函數中,兩者只能選其一。

下面程序很簡單的演示了final的常規用法:

public class Test{

final int t = 1; // 在定義時給值

// 或者(兩者只能選其一)

final int t;

public Test(){

t = 3; // 構造時給值

}

}

還有一種用法是定義方法中的參數爲final,對於基本類型的變量,這樣做並沒有什麼實際意義,因爲基本類型的變量在調用方法時是傳值的,也就是說你可以在方法中更改這個參數變量而不會影響到調用語句,然而對於對象變量,卻顯得很實用,因爲對象變量在傳遞時是傳遞其引用,這樣你在方法中對對象變量的修改也會影響到調用語句中的對象變量,當你在方法中不需要改變作爲參數的對象變量時,明確使用final進行聲明,會防止你無意的修改而影響到調用方法。

另外方法中的內部類在用到方法中的參變量時,此參變也必須聲明爲final纔可使用,如下代碼所示:

public class Test{

void print(final String str){

class InnerTest{

InnerTest (){

System.out.println(str);

}

}

InnerTest it=new InnerTest ();

}

public static void main(String[] args){

Test test=new Test();

test.print("Hello word!!!");

}

}

final方法

將方法聲明爲final那有兩個原因,第一就是說明你已經知道這個方法提供的功能已經滿足你要求,不需要進行擴展,並且也不允許任何從此類繼承的類來覆寫這個方法,但是繼承仍然可以繼承這個方法,也就是說可以直接使用。第二就是允許編譯器將所有對此方法的調用轉化爲inline(行內)調用的機制,它會使你在調用final方法時,直接將方法主體插入到調用處,而不是進行例行的方法調用,例如保存斷點,壓棧等,這樣可能會使你的程序效率有所提高,然而當你的方法主體非常龐大時,或你在多處調用此方法,那麼你的調用主體代碼便會迅速膨脹,可能反而會影響效率,所以你要慎用final進行方法定義。

final類

當你將final用於類身上時,你就需要仔細考慮,因爲一個final類是無法被任何人繼承的,那也就意味着此類在一個繼承樹中是一個葉子類,並且此類的設計已被認爲很完美而不需要進行修改或擴展。對於final類中的成員,你可以定義其爲final,也可以不是final。而對於方法,由於所屬類爲final的關係,自然也就成了final型的。你也可以明確的給final類中的方法加上一個final,但這顯然沒有意義。

finally:

finally 關鍵字是對 Java 異常處理模型的最佳補充。 finally 結構使代碼總會執行,而不管有無異常發生。使用 finally 可以維護對象的內部狀態,並可以清理非內存資源。如果沒有 finally,您的代碼就會很費解。例如,下面的代碼說明,在不使用 finally 的情況下您如何編寫代碼來釋放非內存資源:

public void writeFile(String filePath, String fileName, String args)

throws IOException

{

FileWriter fw = new FileWriter(filePath + fileName);

try {

fw.write(args);

} catch (IOException e) {

//1

fw.close();

throw e;

}

//2

fw.close();

}

這段代碼創建了一個FileWriter object,並調用 write 方法。在退出該方法之前,您必須關閉FileWriter object,以避免資源漏洞。爲了完成這一任務,我們在 //2 處調用 close,它是該方法的最後一條語句。但是,如果 try 塊中發生一個異常會怎麼樣呢?在這種情況下,//2 處的 close 調用永遠不會發生。因此,您必須捕獲這個異常,並在重新發出這個異常之前在 //1 處插入對 close 的另一個調用。這樣就可以確保在退出該方法之前關閉FileWriter object。這樣編寫代碼既麻煩又易於出錯,但在沒有 finally 的情況下這是必不可少的。有了 finally,前面的代碼就可以重寫爲以下的形式:

public void writeFile(String filePath, String fileName, String args)

throws IOException

{

FileWriter fw = new FileWriter(filePath + fileName);

try {

fw.write(args);

} catch (IOException e) {

throw e;

} finally {

fw.close();

}

}

finally 塊確保 close 方法總被執行,而不管 try 塊內是否發出異常。因此,可以確保在退出該方法之前總會調用 close 方法。這樣您就可以確信FileWriter object被關閉並且您沒有泄漏資源。

finalize:

根據Java語言規範,JVM保證調用finalize函數之前,這個對象是不可達的,但是JVM不保證這個函數一定會被調用。另外,規範還保證finalize函數最多運行一次。

通常,finalize用於一些不容易控制、並且非常重要資源的釋放,例如一些I/O的操作,數據的連接。這些資源的釋放對整個應用程序是非常關鍵的。在這種情況下,程序員應該以通過程序本身管理(包括釋放)這些資源爲主,以finalize函數釋放資源方式爲輔,形成一種雙保險的管理機制,而不應該僅僅依靠finalize來釋放資源。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章