單例與序列化

541732025發表於2014-12-03
【轉載】原文見

原文的題目叫“完美的單例實現”,但是看了下內容,不敢苟同,就我所知,要寫一個安全、完美的單例是要考慮很多方面的,譬如懶載入、併發問題、記憶體模型、指令重排等。
但是原文中並有提及到這些,僅僅說了一下序列化與單例,所以這裡我就叫“單例與序列化”吧。
我經常遇到一些這樣的Java程式設計師,他們不確定應該如何恰當的實現單例模式。
我不考慮線上程的環境中合適的實現。但是使用你能在網路上找到的大多數常見的實現方式,你可以輕鬆地建立你想要的多種單例實現。
假設你有下面這種常見的單例的實現:
public final class NonSafeSingleton implements Serializable {
    private static final NonSafeSingleton INSTANCE = new NonSafeSinglet
    private NonSafeSingleton() {}
    public static NonSafeSingleton getInstance(){
        return INSTANCE;    
    }    
}


現在,注意到Serializable 這個單詞。思考一會…..你是對的。如果透過RMI傳送上邊的程式碼,你將會得到第二個例項。它應該足夠可以做一些記憶體中的序列化和反序列化操作。
你剛剛違反了單例的規則。那不是很好。但是如何修復它?通常我會用兩種方式:
1,困難的方式(如果你用4或者更老的版本)
你需要在你的單例類中實現一個readResolve方法。這通常用來重寫序列化機制已經建立的內容。在這個方法裡返回的是用來代替來自序列化的資料。這裡僅需要返回你的例項:
...  
protected Object readResolve() throws ObjectStreamException {      
   return INSTANCE;  
}
...

2,簡單的方式(如果你用5或更新的版本)
將你的單例類改成列舉型別,然後移除私有構造方法和getInstance方法。下面,真的很簡單。然後你將免費得到下面這個:
public enum SafeSingleton implements Serializable {
   INSTANCE;
}

當你再實現單例模式時,記住這些。如果你大量的使用RMI,它可以使你的生活更加簡單。

來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/28912557/viewspace-1355780/,如需轉載,請註明出處,否則將追究法律責任。

相關文章