單例模式

模式定義

單例模式(Singleton Pattern):單例模式確保某一個類只有一個實例嫡良,而且自行實例化并向整個系統(tǒng)提供這個實例倡勇,這個類稱為單例類揪罕,它提供全局訪問的方法糙麦。

單例模式的要點有三個:
一是某個類只能有一個實例;
二是它必須自行創(chuàng)建這個實例牌捷;
三是它必須自行向整個系統(tǒng)提供這個實例墙牌。
單例模式是一種對象創(chuàng)建型模式。單例模式又名單件模式或單態(tài)模式暗甥。

模式結(jié)構(gòu)

類圖
時序圖
image.png
步驟 1
創(chuàng)建一個 Singleton 類喜滨。
SingleObject.java
public class SingleObject {

   //創(chuàng)建 SingleObject 的一個對象
   private static SingleObject instance = new SingleObject();

   //讓構(gòu)造函數(shù)為 private,這樣該類就不會被實例化
   private SingleObject(){}

   //獲取唯一可用的對象
   public static SingleObject getInstance(){
      return instance;
   }

   public void showMessage(){
      System.out.println("Hello World!");
   }
}
步驟 2
從 singleton 類獲取唯一的對象撤防。
SingletonPatternDemo.java
public class SingletonPatternDemo {
   public static void main(String[] args) {

      //不合法的構(gòu)造函數(shù)
      //編譯時錯誤:構(gòu)造函數(shù) SingleObject() 是不可見的
      //SingleObject object = new SingleObject();

      //獲取唯一可用的對象
      SingleObject object = SingleObject.getInstance();

      //顯示消息
      object.showMessage();
   }
}
步驟 3
驗證輸出虽风。
Hello World!

單例模式的幾種實現(xiàn)方式

1、懶漢式寄月,線程不安全
是否 Lazy 初始化:是
是否多線程安全:否
實現(xiàn)難度:易
描述:這種方式是最基本的實現(xiàn)方式辜膝,這種實現(xiàn)最大的問題就是不支持多線程。因為沒有加鎖 synchronized漾肮,所以嚴(yán)格意義上它并不算單例模式厂抖。
這種方式 lazy loading 很明顯,不要求線程安全克懊,在多線程不能正常工作忱辅。
代碼實例:
public class Singleton {  
    private static Singleton instance;  
    private Singleton (){}  
  
    public static Singleton getInstance() {  
    if (instance == null) {  
        instance = new Singleton();  
    }  
    return instance;  
    }  
}  
接下來介紹的幾種實現(xiàn)方式都支持多線程,但是在性能上有所差異谭溉。
2墙懂、懶漢式,線程安全
是否 Lazy 初始化:是
是否多線程安全:是
實現(xiàn)難度:易
描述:這種方式具備很好的 lazy loading扮念,能夠在多線程中很好的工作损搬,但是,效率很低柜与,99% 情況下不需要同步巧勤。
優(yōu)點:第一次調(diào)用才初始化,避免內(nèi)存浪費旅挤。
缺點:必須加鎖 synchronized 才能保證單例踢关,但加鎖會影響效率伞鲫。
getInstance() 的性能對應(yīng)用程序不是很關(guān)鍵(該方法使用不太頻繁)粘茄。
代碼實例:
public class Singleton {  
    private static Singleton instance;  
    private Singleton (){}  
    public static synchronized Singleton getInstance() {  
    if (instance == null) {  
        instance = new Singleton();  
    }  
    return instance;  
    }  
} 
3、餓漢式
是否 Lazy 初始化:否
是否多線程安全:是
實現(xiàn)難度:易
描述:這種方式比較常用,但容易產(chǎn)生垃圾對象柒瓣。
優(yōu)點:沒有加鎖儒搭,執(zhí)行效率會提高。
缺點:類加載時就初始化芙贫,浪費內(nèi)存搂鲫。
它基于 classloder 機制避免了多線程的同步問題,不過磺平,instance 在類裝載時就實例化魂仍,雖然導(dǎo)致類裝載的原因有很多種,在單例模式中大多數(shù)都是調(diào)用 getInstance 方法拣挪, 但是也不能確定有其他的方式(或者其他的靜態(tài)方法)導(dǎo)致類裝載擦酌,這時候初始化 instance 顯然沒有達到 lazy loading 的效果。
代碼實例:
public class Singleton {  
    private static Singleton instance = new Singleton();  
    private Singleton (){}  
    public static Singleton getInstance() {  
    return instance;  
    }  
}  
4菠劝、雙檢鎖/雙重校驗鎖(DCL赊舶,即 double-checked locking)
JDK 版本:JDK1.5 起
是否 Lazy 初始化:是
是否多線程安全:是
實現(xiàn)難度:較復(fù)雜
描述:這種方式采用雙鎖機制,安全且在多線程情況下能保持高性能赶诊。
getInstance() 的性能對應(yīng)用程序很關(guān)鍵笼平。
代碼實例:
public class Singleton {  
    private volatile static Singleton singleton;  
    private Singleton (){}  
    public static Singleton getSingleton() {  
    if (singleton == null) {  
        synchronized (Singleton.class) {  
        if (singleton == null) {  
            singleton = new Singleton();  
        }  
        }  
    }  
    return singleton;  
    }  
}  
5、登記式/靜態(tài)內(nèi)部類
是否 Lazy 初始化:是
是否多線程安全:是
實現(xiàn)難度:一般
描述:這種方式能達到雙檢鎖方式一樣的功效舔痪,但實現(xiàn)更簡單寓调。對靜態(tài)域使用延遲初始化,應(yīng)使用這種方式而不是雙檢鎖方式锄码。這種方式只適用于靜態(tài)域的情況捶牢,雙檢鎖方式可在實例域需要延遲初始化時使用。
這種方式同樣利用了 classloder 機制來保證初始化 instance 時只有一個線程巍耗,它跟第 3 種方式不同的是:第 3 種方式只要 Singleton 類被裝載了秋麸,那么 instance 就會被實例化(沒有達到 lazy loading 效果),而這種方式是 Singleton 類被裝載了炬太,instance 不一定被初始化灸蟆。因為 SingletonHolder 類沒有被主動使用,只有顯示通過調(diào)用 getInstance 方法時亲族,才會顯示裝載 SingletonHolder 類炒考,從而實例化 instance。想象一下霎迫,如果實例化 instance 很消耗資源斋枢,所以想讓它延遲加載,另外一方面知给,又不希望在 Singleton 類加載時就實例化瓤帚,因為不能確保 Singleton 類還可能在其他的地方被主動使用從而被加載描姚,那么這個時候?qū)嵗?instance 顯然是不合適的。這個時候戈次,這種方式相比第 3 種方式就顯得很合理轩勘。
代碼實例:
public class Singleton {  
    private static class SingletonHolder {  
    private static final Singleton INSTANCE = new Singleton();  
    }  
    private Singleton (){}  
    public static final Singleton getInstance() {  
    return SingletonHolder.INSTANCE;  
    }  
}   
6、枚舉
JDK 版本:JDK1.5 起
是否 Lazy 初始化:否
是否多線程安全:是
實現(xiàn)難度:易
描述:這種實現(xiàn)方式還沒有被廣泛采用怯邪,但這是實現(xiàn)單例模式的最佳方法绊寻。它更簡潔,自動支持序列化機制悬秉,絕對防止多次實例化澄步。
這種方式是 Effective Java 作者 Josh Bloch 提倡的方式,它不僅能避免多線程同步問題和泌,而且還自動支持序列化機制驮俗,防止反序列化重新創(chuàng)建新的對象,絕對防止多次實例化允跑。不過王凑,由于 JDK1.5 之后才加入 enum 特性,用這種方式寫不免讓人感覺生疏聋丝,在實際工作中索烹,也很少用。
不能通過 reflection attack 來調(diào)用私有構(gòu)造方法弱睦。
代碼實例:
public enum Singleton {  
    INSTANCE;  
    public void whateverMethod() {  
    }  
}  
經(jīng)驗之談:一般情況下百姓,不建議使用第 1 種和第 2 種懶漢方式,建議使用第 3 種餓漢方式况木。只有在要明確實現(xiàn) lazy loading 效果時垒拢,才會使用第 5 種登記方式。如果涉及到反序列化創(chuàng)建對象時火惊,可以嘗試使用第 6 種枚舉方式求类。如果有其他特殊的需求,可以考慮使用第 4 種雙檢鎖方式屹耐。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末尸疆,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子惶岭,更是在濱河造成了極大的恐慌寿弱,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,284評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件按灶,死亡現(xiàn)場離奇詭異症革,居然都是意外死亡,警方通過查閱死者的電腦和手機鸯旁,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,115評論 3 395
  • 文/潘曉璐 我一進店門噪矛,熙熙樓的掌柜王于貴愁眉苦臉地迎上來量蕊,“玉大人,你說我怎么就攤上這事摩疑。” “怎么了畏铆?”我有些...
    開封第一講書人閱讀 164,614評論 0 354
  • 文/不壞的土叔 我叫張陵雷袋,是天一觀的道長。 經(jīng)常有香客問我辞居,道長楷怒,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,671評論 1 293
  • 正文 為了忘掉前任瓦灶,我火速辦了婚禮鸠删,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘贼陶。我一直安慰自己刃泡,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 67,699評論 6 392
  • 文/花漫 我一把揭開白布碉怔。 她就那樣靜靜地躺著烘贴,像睡著了一般。 火紅的嫁衣襯著肌膚如雪撮胧。 梳的紋絲不亂的頭發(fā)上桨踪,一...
    開封第一講書人閱讀 51,562評論 1 305
  • 那天,我揣著相機與錄音芹啥,去河邊找鬼锻离。 笑死,一個胖子當(dāng)著我的面吹牛墓怀,可吹牛的內(nèi)容都是我干的汽纠。 我是一名探鬼主播,決...
    沈念sama閱讀 40,309評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼傀履,長吁一口氣:“原來是場噩夢啊……” “哼疏虫!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起啤呼,我...
    開封第一講書人閱讀 39,223評論 0 276
  • 序言:老撾萬榮一對情侶失蹤卧秘,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后官扣,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體翅敌,經(jīng)...
    沈念sama閱讀 45,668評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,859評論 3 336
  • 正文 我和宋清朗相戀三年惕蹄,在試婚紗的時候發(fā)現(xiàn)自己被綠了蚯涮。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片治专。...
    茶點故事閱讀 39,981評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖遭顶,靈堂內(nèi)的尸體忽然破棺而出张峰,到底是詐尸還是另有隱情,我是刑警寧澤棒旗,帶...
    沈念sama閱讀 35,705評論 5 347
  • 正文 年R本政府宣布喘批,位于F島的核電站,受9級特大地震影響铣揉,放射性物質(zhì)發(fā)生泄漏饶深。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,310評論 3 330
  • 文/蒙蒙 一逛拱、第九天 我趴在偏房一處隱蔽的房頂上張望敌厘。 院中可真熱鬧,春花似錦朽合、人聲如沸俱两。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,904評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽锋华。三九已至,卻和暖如春箭窜,著一層夾襖步出監(jiān)牢的瞬間毯焕,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,023評論 1 270
  • 我被黑心中介騙來泰國打工磺樱, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留纳猫,地道東北人。 一個月前我還...
    沈念sama閱讀 48,146評論 3 370
  • 正文 我出身青樓竹捉,卻偏偏與公主長得像芜辕,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子块差,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,933評論 2 355

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

  • 1 場景問題# 1.1 讀取配置文件的內(nèi)容## 考慮這樣一個應(yīng)用侵续,讀取配置文件的內(nèi)容。 很多應(yīng)用項目憨闰,都有與應(yīng)用相...
    七寸知架構(gòu)閱讀 6,770評論 12 68
  • 摘要:設(shè)計模式之一:單例模式目錄介紹1.單例模式介紹2.單例模式定義3.單例模式使用場景4.單例模式的實現(xiàn)方式 4...
    肆虐的悲傷閱讀 460評論 0 2
  • 【學(xué)習(xí)難度:★☆☆☆☆状蜗,使用頻率:★★★★☆】直接出處:單例模式梳理和學(xué)習(xí):https://github.com/...
    BruceOuyang閱讀 675評論 1 2
  • 目錄 本文的結(jié)構(gòu)如下: 什么是單例模式 為什么要用該模式 模式的結(jié)構(gòu) 代碼示例 優(yōu)點和缺點 適用環(huán)境 模式應(yīng)用 總...
    w1992wishes閱讀 388評論 1 2
  • 模式動機 對于系統(tǒng)中的某些類來說,只有一個實例很重要鹉动,例如轧坎,一個系統(tǒng)中可以存在多個打印任務(wù),但是只能有一個正在工作...
    lijun_m閱讀 399評論 0 0