[Effective Java] (03)用私有構(gòu)造器或者枚舉類型強化Singleton

Singleton指僅僅被實例化一次的類流礁,常用來代表那些本質(zhì)上唯一的系統(tǒng)組件(窗口管理器或者文件系統(tǒng));

  • 使類成為Singleton會使它的客戶端測試變得十分困難荞膘,因為無法給Singleton替換模擬實現(xiàn)崩泡,除非它實現(xiàn)一個充當(dāng)其類型的接口窟坐。
1. 實現(xiàn)Singleton常用方法
  • 靜態(tài)成員(公有域方法)
public class Singleton {
    public static final Singleton instance = new Singleton();

    private Singleton() { } 
}
  • 靜態(tài)工廠方法
public class Singleton {
    private static final Singleton instance = new Singleton();
    
    private Singleton() { } 
    
    public static Singleton getInstance() {
        return instance;
    }
}

注:該兩種方法并不能保證Singleton的全局唯一性弄息,其可以通過反射機制痊班,設(shè)置AccessibleObject.setAccessible(true),改變構(gòu)造器的訪問屬性,調(diào)用構(gòu)造器生成新的實例摹量;如:

Constructor<?> constructor = Singleton.class.getDeclaredConstructors()[0];
constructor.setAccessible(true);
Singleton instance = (Singleton)constructor.newInstance();

為了抵御這種攻擊涤伐,可以修改構(gòu)造器,讓其在被要求創(chuàng)建實例時拋出異常缨称,如下代碼所示:

public class Singleton {
    private static int count = 0;
    private static final Singleton instance = new Singleton();  
        
    private Singleton() { 
        if(count > 0) {
            throw new IllegalArgumentException("Cannot create Singleton twice!");
        }
        count++;
    }   
    
    public static Singleton getInstance() {
        return instance;
    }
}

注:靜態(tài)變量count一定要在Singleton對象instance前面凝果,否則可以多調(diào)用一次構(gòu)造函數(shù)。

公有域方法與靜態(tài)工廠方法對比如下:

  • 公有域方法主要好處在:這種方法很清晰的表明了這個類是Singleton具钥,公有的靜態(tài)域是final的豆村,所有該域?qū)⒖偘嗤膶ο笠茫?strong>不過公有域在性能上不再有任何優(yōu)勢:現(xiàn)代JVM實現(xiàn)幾乎將靜態(tài)工廠方法的調(diào)用內(nèi)聯(lián)化;
  • 工廠方法的好處:提供了靈活性:可以在不改變其API的前提下骂删,改變該類是否為Singleton的想法。
2. 支持反序列化

僅僅通過implements Serializable支持序列化是不夠的四啰,應(yīng)為每次放序列化一個序列化的實例時宁玫,都會創(chuàng)建一個新的實例,為防止這種情況發(fā)生柑晒,需添加readResolve方法欧瘪,如:

為了維護Singleton,必須聲明所有的實例域都是瞬時(transient)的匙赞,并提供readResolve方法佛掖。

public class Singleton implements Serializable{

    private static int count = 0;
    private static final Singleton instance = new Singleton();  
        
    private Singleton() { 
        if(count > 0) {
            throw new IllegalArgumentException("Cannot create Singleton twice!");
        }
        count++;
        System.out.println(count);
    }   
    
    public static Singleton getInstance() {
        return instance;
    }
    
    
    private Object readResolve() {
        return instance;
    }
    
    public static void main(String[] args) throws IOException, ClassNotFoundException {
        
        Singleton singleton = Singleton.getInstance();
        
        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("Singleton.QyQ"));
        oos.writeObject(singleton);
        oos.close();

        ObjectInputStream ois = new ObjectInputStream(new FileInputStream("Singleton.QyQ"));
        Singleton o = (Singleton) ois.readObject();
        ois.close();
        System.out.println(singleton == o);
    }
}
3. 單個元素枚舉類型實現(xiàn)Singleton
public enum Singleton {

    instance;
    
    public void leaveTheBuilding() {
        System.out.println("Come on baby, QyQiaoo");
    }
    
    public static void main(String[] args) {
        Singleton singleton = Singleton.instance;
        singleton.leaveTheBuilding();
    }
}

該方法在功能上與公有域方法相近,但更加簡潔涌庭,無償提供序列化機制芥被,在面對復(fù)雜的序列化或者反射攻擊的時候也能防止多次實例化;

  • (Enum防止通過反射實例化原因未完待續(xù)··· |)
  • (Enum反序列化機制未完待續(xù)···)

單個元素枚舉類型已經(jīng)成為實現(xiàn)Singleton的最佳方法坐榆。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末拴魄,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌匹中,老刑警劉巖夏漱,帶你破解...
    沈念sama閱讀 206,968評論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異顶捷,居然都是意外死亡挂绰,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,601評論 2 382
  • 文/潘曉璐 我一進店門服赎,熙熙樓的掌柜王于貴愁眉苦臉地迎上來扮授,“玉大人,你說我怎么就攤上這事专肪∩膊” “怎么了?”我有些...
    開封第一講書人閱讀 153,220評論 0 344
  • 文/不壞的土叔 我叫張陵嚎尤,是天一觀的道長荔仁。 經(jīng)常有香客問我,道長芽死,這世上最難降的妖魔是什么乏梁? 我笑而不...
    開封第一講書人閱讀 55,416評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮关贵,結(jié)果婚禮上遇骑,老公的妹妹穿的比我還像新娘。我一直安慰自己揖曾,他們只是感情好落萎,可當(dāng)我...
    茶點故事閱讀 64,425評論 5 374
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著炭剪,像睡著了一般练链。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上奴拦,一...
    開封第一講書人閱讀 49,144評論 1 285
  • 那天媒鼓,我揣著相機與錄音,去河邊找鬼错妖。 笑死绿鸣,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的暂氯。 我是一名探鬼主播潮模,決...
    沈念sama閱讀 38,432評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼株旷!你這毒婦竟也來了再登?” 一聲冷哼從身側(cè)響起尔邓,我...
    開封第一講書人閱讀 37,088評論 0 261
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎锉矢,沒想到半個月后梯嗽,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,586評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡沽损,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,028評論 2 325
  • 正文 我和宋清朗相戀三年灯节,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片绵估。...
    茶點故事閱讀 38,137評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡炎疆,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出国裳,到底是詐尸還是另有隱情形入,我是刑警寧澤,帶...
    沈念sama閱讀 33,783評論 4 324
  • 正文 年R本政府宣布缝左,位于F島的核電站亿遂,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏渺杉。R本人自食惡果不足惜蛇数,卻給世界環(huán)境...
    茶點故事閱讀 39,343評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望是越。 院中可真熱鬧耳舅,春花似錦、人聲如沸倚评。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,333評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽蔓纠。三九已至辑畦,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間腿倚,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,559評論 1 262
  • 我被黑心中介騙來泰國打工蚯妇, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留敷燎,地道東北人。 一個月前我還...
    沈念sama閱讀 45,595評論 2 355
  • 正文 我出身青樓箩言,卻偏偏與公主長得像硬贯,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子陨收,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,901評論 2 345

推薦閱讀更多精彩內(nèi)容