設計模式(2) 單例模式

来源:https://www.cnblogs.com/zhixin9001/archive/2020/07/02/13227527.html
-Advertisement-
Play Games

單例模式 線程安全的Singleton 會破壞Singleton的情況 線程級Singleton 單例模式是幾個創建型模式中最獨立的一個,它的主要目標不是根據客戶程式調用生成一個新的實例,而是控制某個類型的實例數量只有一個。 GOF對單例的描述為: Ensure a class only has o ...


  • 單例模式
  • 線程安全的Singleton
  • 會破壞Singleton的情況
  • 線程級Singleton

單例模式是幾個創建型模式中最獨立的一個,它的主要目標不是根據客戶程式調用生成一個新的實例,而是控制某個類型的實例數量只有一個。
GOF對單例的描述為:
Ensure a class only has one instance, and provide aglobal point of access to.
—Design Patterns : Elements of Reusable Object-Oriented Software

單例模式

單例模式的應用場景不必贅述,先來一個最簡單的實現方式:

public class Singleton
{
    private Singleton() { }
    private static Singleton instance;
    public static Singleton Instance()
    {
        if (instance == null)
        {
            instance = new Singleton();
        }
        return instance;
    }
}

這裡採用的是Lazy方式,也可以在靜態變數被創建的時候直接初始化實例。
這段代碼已經可以滿足最初Singleton模式的設計要求,在大多數情況下可以很好地工作。但在多線程環境下這種實現方式是存在缺陷的,當多個線程幾乎同時調用Singleton類的Instance靜態屬性的時候,instance成員可能還沒有被實例化,因此它被創建了多次,而且最終Singleton類中保存的是最後創建的那個實例,各個線程引用的對象不同。

線程安全的Singleton

為了保證多線程環境下instance實例只有一個,對代碼進行了優化:

public class Singleton
{
    private static volatile Singleton instance;
    public static Singleton Instance()
    {
        if (instance == null)
        {
            lock (typeof(Singleton))
            {
                if (instance == null)
                {
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}

相比最初的實現,改變的地方有這幾處:

  • instance使用volatile關鍵字修飾,它表示欄位可能被多個併發執行的線程修改。
  • 在實例化前lock Singleton類型,避免了多個線程同時實例化的問題。
  • 第一個if加在了lock之前,是為了避免每次調用都鎖定Singleton類型帶來的效率下降。
  • lock後再次判斷instance是否為空,是因為在高併發場景下,在第一個線程鎖定並實例化期間,仍然可能會有別的線程進入到第一層if內,這樣如果不再次判空,就會重覆實例化。

會破壞Singleton的情況

有些情況會破壞Singleton的封裝,跳過“只能有一個實例”的限制,在實際應用中要註意規避。

  • 第一種情況就是實現ICloneable介面或繼承自其相關的子類,這樣客戶程式藉助ICloneable介面就可以跳過已經被隱藏起來的構造函數

  • 另外通過二進位、Json之類序列化、反序列化的方式也可以產生新的對象。

線程級Singleton

前面討論的是線程安全的Singleton實現,但有時需要的是更細粒度的Singleton,比如線程級的Singleton,只要保證在一個線程內只有一個實例即可,這就類似Asp.NET Core 自帶的IOC提供的AddScope註冊方式,可以保證一個HttpContext內只有一個實例。

雖然Asp.NET Core提供類似的現成實現,但如果在非Web環境下也需要線程級的實例控制該怎麼辦呢? 結合C#提供的System.ThreadStaticAttribute可以完成

通過System.ThreadStaticAttribute可以將某個靜態變數限定為僅在本線程內部是靜態的。
實現如下:

public class ThreadSingleton
{
    private ThreadSingleton() { }

    [ThreadStatic] //instance只在當前線程內為靜態
    private static ThreadSingleton instance;
    public static ThreadSingleton Instance()
    {
        if (instance == null)
        {
            instance = new ThreadSingleton();
        }
        return instance;
    }
}

這裡再不需要線程鎖了,因為線程級的單例不需要考慮線程安全。
為了驗證實現的準確性,首先構造一個線程內執行的目標對象:

class Work
{
    public static IList<int> Log = new List<int>();

    /// <summary>
    /// 每個線程的執行部分
    /// </summary>
    public void Procedure()
    {
        ThreadSingleton s1 = ThreadSingleton.Instance();
        ThreadSingleton s2 = ThreadSingleton.Instance();

        //證明可以正常構造實例
        Assert.IsNotNull(s1);
        Assert.IsNotNull(s2);

        //驗證當前線程執行體內兩次獲取的是同一個實例
        Assert.AreEqual(s1.GetHashCode(), s2.GetHashCode());

        //記錄當前線程所使用對象的HashCode
        Log.Add(s1.GetHashCode());
    }
}

這個類會在每個線程內部執行,並驗證線程內多次獲取的Instance是同一個實例,並記錄這個實例的HashCode,以便與別的線程實例對比。
接下來開啟多個線程同時執行Procedure()方法:

[Test]
public void ThreadSingletonTest()
{
    int threadCount = 4;
    Thread[] threads = new Thread[threadCount];  //創建4個線程
    for (int i = 0; i < threadCount; i++)
    {
        ThreadStart work = new ThreadStart(new Work().Procedure);
        threads[i] = new Thread(work);
    }

    //執行線程
    foreach (var thread in threads)
    {
        thread.Start();
    }

    Thread.Sleep(10000);
    Assert.AreEqual(threadCount, Work.Log.Distinct().Count());
}

Work類的靜態變數Log中記錄了每個線程中實例的HashCode,這些HashCode彼此不相同,且與線程的數量一致,證明每個線程間的實例是不相同的。

參考書籍:
王翔著 《設計模式——基於C#的工程化實現及擴展》


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 憑藉應用廣泛、入門簡單的優勢,Web前端吸引了人們的廣泛關註。學習Web前端就業薪資高,因此很多人都想入門前端開發行業。 零基礎自學Web前端,你需要具備以下幾點: 1、耐性。要成為優秀的web前端開發者,要調整好心態。拋開一切的方法和技術知識,最重要的就是你的耐性。 2、學會延伸。對於新手來說,新 ...
  • 目錄判斷 js 類型的方式ES5 和 ES6 分別幾種方式聲明變數閉包的概念?優缺點?淺拷貝和深拷貝數組去重的方法DOM 事件有哪些階段?談談對事件代理的理解js 執行機制、事件迴圈介紹下 promise.allasync 和 awaitES6 的 class 和構造函數的區別transform、t ...
  • 不知道大家有沒有註意過對象中的一些通用方法,例如所有所有的對象都有 toString、constructor 等等一些方法。 當然如果要仔細看的話,大家可以: var a = {}; console.log(a); 我們可以清晰的看到他有很多的內置方法。當然,也可以看到最下麵有兩個比較怪的方法 ge ...
  • 應屆生:阿姨,我不想努力了在學校用React + antd做過後臺管理系統,熟悉React技術棧。兩年前端:公司技術棧是React,都用了一年了,我React賊六。五年前端:帶團隊把公司的糞坑項目用React重構了。React對我來說就跟呼吸一樣容易。:要不學學React源碼吧。......%……& ...
  • 1.HTML的學習內容 1.HTML的概念 2.互聯網的三大基石 3.HTML的頭標簽 4.HTML的主體標簽 5.HTML的圖片標簽 6.HTML的超鏈接標簽7.HTML的表格標簽 8.HTML的內嵌和框架標簽 9.HTML的表單 2.HTML的概念 HTML:超文本標記語言 作用:需要將java ...
  • 在前面隨筆《循序漸進VUE+Element 前端應用開發(12)--- 整合ABP框架的前端登錄處理》簡單的介紹了一個結合ABP後端的登陸介面實現前端系統登陸的功能,本篇隨筆繼續深化這一主題,著重介紹基於ABP後端介面信息,實現對前端界面的開發工作。 ABP(ASP.NET Boilerplate)... ...
  • 以下麵試題來自騰訊、阿裡、網易、餓了麽、美團、拼多多、百度等等大廠綜合起來常考的題目。 如何寫一個漂亮的簡歷 簡歷不是一份記流水賬的東西,而是讓用人方瞭解你的亮點的。平時有在做一些修改簡歷的收費服務,也算看過蠻多簡歷了。很多簡歷都有如下特征 喜歡說自己的特長、優點,用人方真的不關註你的性格是否陽光等 ...
  • 從webpack打包結構中我們知道,vue中有一個存放外部資源的文件夾static,它裡面的文件是不會被打包編譯的,所以我們就可以利用外部引入js的方式將我們的想要的數據在index.html中以js文件的方式引入,然後就可以全局使用。 具體的方法如下: 1.在項目中找到static文件夾,在裡面創 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...