淺談C#單例模式的實(shí)現(xiàn)和性能對(duì)比
簡(jiǎn)介
單例指的是只能存在一個(gè)實(shí)例的類(在C#中,更準(zhǔn)確的說(shuō)法是在每個(gè)AppDomain之中只能存在一個(gè)實(shí)例的類,它是軟件工程中使用最多的幾種模式之一。在第一個(gè)使用者創(chuàng)建了這個(gè)類的實(shí)例之后,其后需要使用這個(gè)類的就只能使用之前創(chuàng)建的實(shí)例,無(wú)法再創(chuàng)建一個(gè)新的實(shí)例。通常情況下,單例會(huì)在第一次被使用時(shí)創(chuàng)建。本文會(huì)對(duì)C#中幾種單例的實(shí)現(xiàn)方式進(jìn)行介紹,并分析它們之間的線程安全性和性能差異。
單例的實(shí)現(xiàn)方式有很多種,但從最簡(jiǎn)單的實(shí)現(xiàn)(非延遲加載,非線程安全,效率低下),到可延遲加載,線程安全,且高效的實(shí)現(xiàn),它們都有一些基本的共同點(diǎn):
- 單例類都只有一個(gè)private的無(wú)參構(gòu)造函數(shù)
 - 類聲明為sealed(不是必須的)
 - 類中有一個(gè)靜態(tài)變量保存著所創(chuàng)建的實(shí)例的引用
 - 單例類會(huì)提供一個(gè)靜態(tài)方法或?qū)傩詠?lái)返回創(chuàng)建的實(shí)例的引用(eg.GetInstance)
 
幾種實(shí)現(xiàn)
一非線程安全
//Bad code! Do not use!
public sealed class Singleton
{
  private static Singleton instance = null;
  private Singleton()
  {
  }
  public static Singleton instance
  {
    get
    {
      if (instance == null)
      {
        instance = new Singleton();
      }
      return instance;
    }
  }
}
這種方法不是線程安全的,會(huì)存在兩個(gè)線程同時(shí)執(zhí)行if (instance == null)并且創(chuàng)建兩個(gè)不同的instance,后創(chuàng)建的會(huì)替換掉新創(chuàng)建的,導(dǎo)致之前拿到的reference為空。
二簡(jiǎn)單的線程安全實(shí)現(xiàn)
public sealed class Singleton
{
  private static Singleton instance = null;
  private static readonly object padlock = new object();
  Singleton()
  {
  }
  public static Singleton Instance
  {
    get
    {
      lock (padlock)
      {
        if (instance == null)
        {
          instance = new Singleton();
        }
        return instance;
      }
    }
  }
}
相比較于實(shí)現(xiàn)一,這個(gè)版本加上了一個(gè)對(duì)instance的鎖,在調(diào)用instance之前要先對(duì)padlock上鎖,這樣就避免了實(shí)現(xiàn)一中的線程沖突,該實(shí)現(xiàn)自始至終只會(huì)創(chuàng)建一個(gè)instance了。但是,由于每次調(diào)用Instance都會(huì)使用到鎖,而調(diào)用鎖的開(kāi)銷較大,這個(gè)實(shí)現(xiàn)會(huì)有一定的性能損失。
注意這里我們使用的是新建一個(gè)private的object實(shí)例padlock來(lái)實(shí)現(xiàn)鎖操作,而不是直接對(duì)Singleton進(jìn)行上鎖。直接對(duì)類型上鎖會(huì)出現(xiàn)潛在的風(fēng)險(xiǎn),因?yàn)檫@個(gè)類型是public的,所以理論上它會(huì)在任何code里調(diào)用,直接對(duì)它上鎖會(huì)導(dǎo)致性能問(wèn)題,甚至?xí)霈F(xiàn)死鎖情況。
Note: C#中,同一個(gè)線程是可以對(duì)一個(gè)object進(jìn)行多次上鎖的,但是不同線程之間如果同時(shí)上鎖,就可能會(huì)出現(xiàn)線程等待,或者嚴(yán)重的會(huì)出現(xiàn)死鎖情況。因此,我們?cè)谑褂胠ock時(shí),盡量選擇類中的私有變量上鎖,這樣可以避免上述情況發(fā)生。
三雙重驗(yàn)證的線程安全實(shí)現(xiàn)
public sealed calss Singleton
{
  private static Singleton instance = null;
  private static readonly object padlock = new object();
  Singleton()
  {
  }
  public static Singleton Instance
  {
    get
    {
      if (instance == null)
      {
        lock (padlock)
        {
          if (instance == null)
          {
            instance = new Singleton();
          }
        }
      }
      return instance;
    }
  } 
}
在保證線程安全的同時(shí),這個(gè)實(shí)現(xiàn)還避免了每次調(diào)用Instance都進(jìn)行l(wèi)ock操作,這會(huì)節(jié)約一定的時(shí)間。
但是,這種實(shí)現(xiàn)也有它的缺點(diǎn):
1無(wú)法在Java中工作。(具體原因可以見(jiàn)原文,這邊沒(méi)怎么理解)
2程序員在自己實(shí)現(xiàn)時(shí)很容易出錯(cuò)。如果對(duì)這個(gè)模式的代碼進(jìn)行自己的修改,要倍加小心,因?yàn)閐ouble check的邏輯較為復(fù)雜,很容易出現(xiàn)思考不周而出錯(cuò)的情況。
四不用鎖的線程安全實(shí)現(xiàn)
public sealed class Singleton
{
  //在Singleton第一次被調(diào)用時(shí)會(huì)執(zhí)行instance的初始化
  private static readonly Singleton instance = new Singleton();
  //Explicit static consturctor to tell C# compiler 
  //not to mark type as beforefieldinit
  static Singleton()
  {
  }
  private Singleton()
  {
  }
  public static Singleton Instance
  {
    get
    {
      return instance;
    }
  }
}
這個(gè)實(shí)現(xiàn)很簡(jiǎn)單,并沒(méi)有用到鎖,但是它仍然是線程安全的。這里使用了一個(gè)static,readonly的Singleton實(shí)例,它會(huì)在Singleton第一次被調(diào)用的時(shí)候新建一個(gè)instance,這里新建時(shí)候的線程安全保障是由.NET直接控制的,我們可以認(rèn)為它是一個(gè)原子操作,并且在一個(gè)AppDomaing中它只會(huì)被創(chuàng)建一次。
這種實(shí)現(xiàn)也有一些缺點(diǎn):
1instance被創(chuàng)建的時(shí)機(jī)不明,任何對(duì)Singleton的調(diào)用都會(huì)提前創(chuàng)建instance
2static構(gòu)造函數(shù)的循環(huán)調(diào)用。如有A,B兩個(gè)類,A的靜態(tài)構(gòu)造函數(shù)中調(diào)用了B,而B(niǎo)的靜態(tài)構(gòu)造函數(shù)中又調(diào)用了A,這兩個(gè)就會(huì)形成一個(gè)循環(huán)調(diào)用,嚴(yán)重的會(huì)導(dǎo)致程序崩潰。
3我們需要手動(dòng)添加Singleton的靜態(tài)構(gòu)造函數(shù)來(lái)確保Singleton類型不會(huì)被自動(dòng)加上beforefieldinit這個(gè)Attribute,以此來(lái)確保instance會(huì)在第一次調(diào)用Singleton時(shí)才被創(chuàng)建。
4readonly的屬性無(wú)法在運(yùn)行時(shí)改變,如果我們需要在程序運(yùn)行時(shí)dispose這個(gè)instance再重新創(chuàng)建一個(gè)新的instance,這種實(shí)現(xiàn)方法就無(wú)法滿足。
五完全延遲加載實(shí)現(xiàn)(fully lazy instantiation)
public sealed class Singleton
{
  private Singleton()
  {
  }
  public static Singleton Instance 
  {
    get
    {
      return Nested.instance;
    }
  }
  private class Nested
  {
    // Explicit static constructor to tell C# compiler
    // not to mark type as beforefieldinit
    static Nested()
    {
    }
    internal static readonly Singleton instance = new Singleton();
  }
}
實(shí)現(xiàn)五是實(shí)現(xiàn)四的包裝。它確保了instance只會(huì)在Instance的get方法里面調(diào)用,且只會(huì)在第一次調(diào)用前初始化。它是實(shí)現(xiàn)四的確保延遲加載的版本。
六 使用.NET4的Lazy<T>類型
public sealed class Singleton
{
  private static readonly Lazy<Singleton> lazy = new Lazy<Singleton>(() => new Singleton());
  public static Singleton Instance 
  {
    get 
    {
      return lazy.Value;
    }
  }
  private Singleton()
  {
  }
}
.NET4或以上的版本支持Lazy<T>來(lái)實(shí)現(xiàn)延遲加載,它用最簡(jiǎn)潔的代碼保證了單例的線程安全和延遲加載特性。
性能差異
之前的實(shí)現(xiàn)中,我們都在強(qiáng)調(diào)代碼的線程安全性和延遲加載。然而在實(shí)際使用中,如果你的單例類的初始化不是一個(gè)很耗時(shí)的操作或者初始化順序不會(huì)導(dǎo)致bug,延遲初始化是一個(gè)可有可無(wú)的特性,因?yàn)槌跏蓟加玫臅r(shí)間是可以忽略不計(jì)的。
在實(shí)際使用場(chǎng)景中,如果你的單例實(shí)例會(huì)被頻繁得調(diào)用(如在一個(gè)循環(huán)中),那么為了保證線程安全而帶來(lái)的性能消耗是更值得關(guān)注的地方。
為了比較這幾種實(shí)現(xiàn)的性能,我做了一個(gè)小測(cè)試,循環(huán)拿這些實(shí)現(xiàn)中的單例9億次,每次調(diào)用instance的方法執(zhí)行一個(gè)count++操作,每隔一百萬(wàn)輸出一次,運(yùn)行環(huán)境是MBP上的Visual Studio for Mac。結(jié)果如下:
| 線程安全性 | 延遲加載 | 測(cè)試運(yùn)行時(shí)間(ms) | |
|---|---|---|---|
| 實(shí)現(xiàn)一 | 否 | 是 | 15532 | 
| 實(shí)現(xiàn)二 | 是 | 是 | 45803 | 
| 實(shí)現(xiàn)三 | 是 | 是 | 15953 | 
| 實(shí)現(xiàn)四 | 是 | 不完全 | 14572 | 
| 實(shí)現(xiàn)五 | 是 | 是 | 14295 | 
| 實(shí)現(xiàn)六 | 是 | 是 | 22875 | 
測(cè)試方法并不嚴(yán)謹(jǐn),但是仍然可以看出,方法二由于每次都需要調(diào)用lock,是最耗時(shí)的,幾乎是其他幾個(gè)的三倍。排第二的則是使用.NET Lazy類型的實(shí)現(xiàn),比其他多了二分之一左右。其余的四個(gè),則沒(méi)有明顯區(qū)別。
總結(jié)
總體來(lái)說(shuō),上面說(shuō)的多種單例實(shí)現(xiàn)方式在現(xiàn)今的計(jì)算機(jī)性能下差距都不大,除非你需要特別大并發(fā)量的調(diào)用instance,才會(huì)需要去考慮鎖的性能問(wèn)題。
對(duì)于一般的開(kāi)發(fā)者來(lái)說(shuō),使用方法二或者方法六來(lái)實(shí)現(xiàn)單例已經(jīng)是足夠好的了,方法四和五則需要對(duì)C#運(yùn)行流程有一個(gè)較好的認(rèn)識(shí),并且實(shí)現(xiàn)時(shí)需要掌握一定技巧,并且他們節(jié)省的時(shí)間仍然是有限的。
引用
本文大部分是翻譯自Implementing the Singleton Pattern in C#,加上了一部分自己的理解。這是我搜索static readonly field initializer vs static constructor initialization時(shí)看到的,在這里對(duì)兩位作者表示感謝。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持我們。
上一篇:C#實(shí)現(xiàn)的簡(jiǎn)單整數(shù)四則運(yùn)算計(jì)算器功能示例
欄 目:C#教程
下一篇:C# 使用Free Spire.Presentation 實(shí)現(xiàn)對(duì)PPT插入、編輯、刪除表格
本文標(biāo)題:淺談C#單例模式的實(shí)現(xiàn)和性能對(duì)比
本文地址:http://www.jygsgssxh.com/a1/C_jiaocheng/5459.html
您可能感興趣的文章
- 01-10C#使用Dispose模式實(shí)現(xiàn)手動(dòng)對(duì)資源的釋放
 - 01-10深入淺出23種設(shè)計(jì)模式
 - 01-10C#正則表達(dá)式的6個(gè)簡(jiǎn)單例子
 - 01-10Python設(shè)計(jì)模式編程中的備忘錄模式與對(duì)象池模式示例
 - 01-10淺談C#中簡(jiǎn)單的異常引發(fā)與處理操作
 - 01-10淺談C#指針問(wèn)題
 - 01-10詳解C#的設(shè)計(jì)模式編程之抽象工廠模式的應(yīng)用
 - 01-10解析C#設(shè)計(jì)模式編程中的裝飾者模式
 - 01-10簡(jiǎn)單了解C#設(shè)計(jì)模式編程中的橋接模式
 - 01-10C#編程中使用設(shè)計(jì)模式中的原型模式的實(shí)例講解
 


閱讀排行
- 1C語(yǔ)言 while語(yǔ)句的用法詳解
 - 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹(shù)的示例代碼(圣誕
 - 3利用C語(yǔ)言實(shí)現(xiàn)“百馬百擔(dān)”問(wèn)題方法
 - 4C語(yǔ)言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
 - 5c語(yǔ)言計(jì)算三角形面積代碼
 - 6什么是 WSH(腳本宿主)的詳細(xì)解釋
 - 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
 - 8正則表達(dá)式匹配各種特殊字符
 - 9C語(yǔ)言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
 - 10C語(yǔ)言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
 
本欄相關(guān)
- 01-10C#通過(guò)反射獲取當(dāng)前工程中所有窗體并
 - 01-10關(guān)于ASP網(wǎng)頁(yè)無(wú)法打開(kāi)的解決方案
 - 01-10WinForm限制窗體不能移到屏幕外的方法
 - 01-10WinForm繪制圓角的方法
 - 01-10C#實(shí)現(xiàn)txt定位指定行完整實(shí)例
 - 01-10WinForm實(shí)現(xiàn)仿視頻播放器左下角滾動(dòng)新
 - 01-10C#停止線程的方法
 - 01-10C#實(shí)現(xiàn)清空回收站的方法
 - 01-10C#通過(guò)重寫(xiě)Panel改變邊框顏色與寬度的
 - 01-10C#實(shí)現(xiàn)讀取注冊(cè)表監(jiān)控當(dāng)前操作系統(tǒng)已
 
隨機(jī)閱讀
- 01-10delphi制作wav文件的方法
 - 04-02jquery與jsp,用jquery
 - 01-11ajax實(shí)現(xiàn)頁(yè)面的局部加載
 - 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
 - 01-10SublimeText編譯C開(kāi)發(fā)環(huán)境設(shè)置
 - 01-10C#中split用法實(shí)例總結(jié)
 - 01-11Mac OSX 打開(kāi)原生自帶讀寫(xiě)NTFS功能(圖文
 - 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
 - 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
 - 01-10使用C語(yǔ)言求解撲克牌的順子及n個(gè)骰子
 


