[創建型模式]單例模式(Singleton)

  • 一個類只能生成一個對象,所有對象對它的依賴都是相同的。
  • 對象的生產是通過 new關鍵字完成的(當然也有其他的方式,如對象復制,反射等)。
  • 使用new關鍵字創建對象時,都會根據輸入的參數調用相應的構造函數,如果我們把構造函數設置為private私有訪問權限時就可以禁止外部創建對象了。
臣子叩拜皇帝類圖

代碼清單

//皇帝類
public class Emperor {
  private static final Emperor emperor = new Emperor();
  //私有的構造函數
  private Emperor(){

  }
  //創建方法
  public static Emperor getInstance(){
    return emperor;
  }

  public void say(){
    system.out.println("My is Emperor");
  }
}

通過定義一個私有訪問權限的構造函數,避免被其他類new出一個對象,而Emperor自己則可以new一個對象出來,其他類對該類的訪問都可以通過getInstance獲得同一個對象。

//臣子類
public class Minister {
  public static void main(String [] args) {
    for(int day=0; day<3 ; day++){
      Emperor emperor = Emperor.getInstance();
      emperor.say();
    }
  }
}

單例模式定義

  • 確保某一個類只有一個實例,而且自行實例化并向整個系統提供這個實例。

單例模式的優點

  • 由于單例模式在內存中只有一個實例,減少了內存開支,特別是一個對象需要頻繁地創建、銷毀時,而且創建或銷毀時性能又無法優化,單例模式的優勢就非常明顯。
  • 由于單例模式只生成一個實例,所以減少了系統的性能開銷,當一個對象的產生需要比較多的資源時,如讀取配置、產生其他依賴對象時,則可以通過在應用啟動時直接產生一個單例對象,然后用永久駐留內存的方式來解決。
  • 單例模式可以避免對資源的多重占用,例如一個寫文件動作,由于只有一個實例存在內存中,避免對同一個資源文件的同時寫操作。
  • 單例模式可以在系統設置全局的訪問點,優化和共享資源訪問,例如可以設計一個單例類,負責所有數據表的映射處理。

單例模式的缺點

  • 單例模式一般沒有接口,擴展很困難,若要擴展,除了修改代碼基本上沒有第二種途徑可以實現。單例模式為什么不能增加接口呢?因為接口對單例模式是沒有任何意義的,它要求自行實例化,并且提供單一實例、接口或抽象類是不可能被實例化的。當然,在特殊的情況下,單例模式可以實現接口、被繼承等,需要在系統開發中根據環境判斷。
  • 單例模式對測試是不利的,在并行開發環境中,如果單例模式沒有完成,是不能進行測試的,沒有接口也不能使用mock的方式虛擬一個對象。
  • 單例模式與單一職責原則有沖突。一個類應該只實現一個邏輯,而不關心它是否是單例的,是不是要單例取決于環境,單例模式把要單例和業務邏輯融合在一個類中。

使用場景

在一個系統中,要求一個類有且僅有一個對象,如果出現多個對象就會出現不良反應,可以采用單例模式。

  • 要求生成唯一序列號的環境。
  • 在整個項目中需要一個共享訪問點或共享數據,例如一個Web頁面上的計數器,可以不用把每次刷新都記錄到數據庫中,使用單例模式保持計數器的值,并確保是線程安全的。
  • 創建一個對象需要消耗的資源過多,如要訪問IO和數據庫等資源。
  • 需要定義大量的靜態常量和靜態方法的環境,可以采用單例模式。

注意事項

首先,在高并發情況下,請注意單例模式的線程同步問題。之前的例子不會出現產生多個實例的情況,但是下面的例子就需要考慮線程同步。

?public class Singleton {
  private static Singleton singleton = null;
  private Singleton() {

  }
  public static Singleton getInstance() {
    if(singleton == null){
      singleton = new Singleton();
    }
    return singleton;
  }
}

該單例模式在低并發的情況下尚不會出現問題,若系統壓力增大,并發量增加時則可能在內存中出現多個實例,破壞了最初預期。為什么會出現這種情況呢?如果一個線程A執行到singleton = new Singleton(),但還沒有獲得對象(對象初始化是需要時間的),第二個線程B也在執行,執行到if(singleton == null)判斷,那么線程B獲得判斷條件也是為真,于是內存中就出現兩個對象。

解決線程不安全的方法有很多,可以在getInstance方法前加synchronized關鍵字,也可以在getInstance方法內增加synchronized來實現。

其次,需要考慮對象的復制情況。在Java中,對象默認是不可以被復制的,若實現了Cloneable接口,并實現了clone方法,則可以直接通過對象復制方式創建一個新對象,對象復制是不用調用類的構造函數,因此即使是私有的構造函數,對象仍然可以被復制。在一般情況下,類復制的情況不需要考慮,很少會出現一個單例類會主動要求被復制的情況,解決該問題最好的方法就是單例類不要實現Cloneable接口。

單例模式的擴展

需要產生固定數量對象的模式就叫做有上限的多例模式,它是單例模式的一種擴展,采用有上限的多例模式,我們可以在設計時決定在內存中有多少個實例,方便系統進行擴展,修正單例可能存在的性能問題,提供系統的響應速度。

//皇帝類
public class Emperor {
  private static int maxNumOfEmperor = 2;
  private static ArrayList<String> nameList = new ArrayList<String>();
  private static ArrayList<Emperor> emperorList = new ArrayList<Emperor>();
  private static int countNumOfEmperor = 0;
  static {
    for(int i=0; i<maxNumOfEmperor; i++){
      emperorList.add(new Emperor("皇"+(i+1)+"帝"));
    }
  }
  //私有的構造函數
  private Emperor(){

  }
  private Emperor(String name){
    nameList.add(name);
  }
  //創建方法
  public static Emperor getInstance(){
    Random random = new Random();
    countNumOfEmperor = random.nextInt(maxNumOfEmperor);
    return emperorList.get(countNumOfEmperor);
  }

  public void say(){
    system.out.println("My is Emperor");
  }
}
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,936評論 6 535
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,744評論 3 421
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,879評論 0 381
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,181評論 1 315
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,935評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,325評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,384評論 3 443
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,534評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,084評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,892評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,067評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,623評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,322評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,735評論 0 27
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,990評論 1 289
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,800評論 3 395
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,084評論 2 375

推薦閱讀更多精彩內容

  • 1 場景問題# 1.1 讀取配置文件的內容## 考慮這樣一個應用,讀取配置文件的內容。 很多應用項目,都有與應用相...
    七寸知架構閱讀 6,854評論 12 68
  • 前言 本文主要參考 那些年,我們一起寫過的“單例模式”。 何為單例模式? 顧名思義,單例模式就是保證一個類僅有一個...
    tandeneck閱讀 2,529評論 1 8
  • Spring Cloud為開發人員提供了快速構建分布式系統中一些常見模式的工具(例如配置管理,服務發現,斷路器,智...
    卡卡羅2017閱讀 134,799評論 18 139
  • 單例模式(SingletonPattern)一般被認為是最簡單、最易理解的設計模式,也因為它的簡潔易懂,是項目中最...
    成熱了閱讀 4,281評論 4 34
  • 我的大學同學念文與班上的安真絕交了,跟我們也漸漸疏遠了,原因我們大家都心知肚明。現在想想,大家還是不勝唏噓。 那一...
    沙小沙閱讀 1,475評論 0 6