單例模式的介紹與實現

来源:https://www.cnblogs.com/ChallengeEverything/archive/2018/07/22/9350024.html
-Advertisement-
Play Games

介紹單例模式之前我們先來介紹下什麼是設計模式,所謂設計模式簡單來說就是根據開發者先輩們的經驗而總結出的解決問題的方式,可以說是前人經驗和心血的體現。 有了設計模式之後,我們可以少走很多彎路,利用設計模式來輕鬆解決對應的問題。 話不多說,今天先來介紹最容易入門和掌握的設計模式——單例模式 單例模式:我 ...


介紹單例模式之前我們先來介紹下什麼是設計模式,所謂設計模式簡單來說就是根據開發者先輩們的經驗而總結出的解決問題的方式,可以說是前人經驗和心血的體現。

有了設計模式之後,我們可以少走很多彎路,利用設計模式來輕鬆解決對應的問題。

話不多說,今天先來介紹最容易入門和掌握的設計模式——單例模式

單例模式:我們知道,在oo語言中,例如c#, JAVA,我們想要創建對象只要通過關鍵字new 即可,那麼如果我們希望整個程式運行過程中某個類只有一個實例該怎麼實現?

這個時候單例模式就出現了,它的目的就是確保程式運行過程中某個類只能有一個實例,並且提供一個全局訪問點。

隨著開發語言的演變,開發環境以及需求的多樣性,單例模式也有了眾多的實現方式,接下來會簡單的介紹幾個。

1. 經典實現

通過將構造函數設置為私有,以及暴露出一個靜態方法,來實現單例模式。

public class SingletonPattern
    {
        private SingletonPattern(){}
        private static SingletonPattern uniqueInstance;

        public static SingletonPattern getInstance() 
        {
            if (uniqueInstance == null)
            {
                uniqueInstance = new SingletonPattern();
            }
            return uniqueInstance;
        }
    }
var test = SingletonPattern.getInstance();
var test2 = SingletonPattern.getInstance();
var test3 = SingletonPattern.getInstance();
Console.WriteLine(test == test2);
Console.WriteLine(test == test3);
Console.WriteLine(test2 == test3);

var item1 = new NormalClass();
var item2 = new NormalClass();
Console.WriteLine(item1 == item2);

由上述代碼可見,通過靜態方法來返回唯一的實例,並且只有在初次調用這個靜態方法的時候才實例化(延遲實例化),這樣也就確保了再需要的時候實例化而不是在程式運行開始實例化。

那麼上述代碼是否已經達到我們的目的了呢?答案是不完全能夠,如果程式運行中確保不會有多個線程同時訪問這個靜態方法,那麼這個寫法已經足夠了,但是如果有呢?

   public class SingletonPattern
    {
        private SingletonPattern(){}
        private static SingletonPattern uniqueInstance;

        public static SingletonPattern getInstance()
        {
            if (uniqueInstance == null)
            {
                Console.WriteLine("實例化");//標記實例化
                uniqueInstance = new SingletonPattern();
            }
            return uniqueInstance;
        }
    }
TaskFactory taskFactory = new TaskFactory();
            for (var i = 0; i <= 5; i++) 
            {
                taskFactory.StartNew(() => SingletonPattern.getInstance());
            }

可以看到多線程下,該類被實例化了四次,那麼該如何解決多線程的問題呢?

2. c#中通過lock關鍵字來解決線程同步問題

只要在上述代碼中略微做些改動,就能夠解決多線程的隱患

public class SingletonPattern
    {
        private SingletonPattern(){}
        private static SingletonPattern uniqueInstance;
        private static object locker = new object();//用於作為lock的對象,一個標誌位,lock對象必須是引用類型

        public static SingletonPattern getInstance()
        {
            lock (locker)//加鎖,locker對象,以及代碼塊中的代碼都不能被其他線程調用
            {
                if (uniqueInstance == null)
                {
                    Console.WriteLine("實例化");//標記實例化
                    uniqueInstance = new SingletonPattern();
                }
                return uniqueInstance;
            }
        }
    }

 

加了鎖之後,我們確實解決了線程同步的問題,但是加鎖對性能有著影響,而且仔細觀察上述代碼,我們對每一個線程都進行了加鎖,是否有這個必要呢?

其實我們加鎖的目的就是為了確保只有一個線程來實例化了該類,一旦初始化之後,之後無論有幾個線程同時調用這個靜態方法,那麼它們都將獲得同一個對象。

 

3. 雙重鎖 if+lock

我們在上述代碼中,做如下改動,確保只有在第一次實例化的時候,對線程加鎖。

public class SingletonPattern
    {
        private SingletonPattern(){}
        private static SingletonPattern uniqueInstance;
        private static object locker = new object();//用於作為lock的對象,一個標誌位,lock對象必須是引用類型

        public static SingletonPattern getInstance()
        {
            if (uniqueInstance == null)//判斷是否已經被實例化,如果沒有,往下走
            {
                lock (locker)//因為沒有被實例化,所以加鎖,接著實例化
                {
                    if (uniqueInstance == null)
                    {
                        Console.WriteLine("實例化");//標記實例化
                        uniqueInstance = new SingletonPattern();
                    }
                }
            }
            return uniqueInstance;
        }
    }

上述代碼執行過程為先判斷是否已經實例化,沒有那麼加鎖創建實例,如果有則返回,一旦創建之後,就沒有必要在去加鎖。

這樣既保證了多線程安全,又沒有對性能造成過多的影響。

 

4. 除了上述幾種方式之外,還可以通過把實例賦值給靜態變數,那麼也能確保該實例唯一,但是這個實例會在程式運行的時候就被初始化,而不是按需初始化。

實現方式不再這邊描述,可以自行嘗試。


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

-Advertisement-
Play Games
更多相關文章
  • 正因為如此,有不少同學使用 Sass 新的語法規則,而文件擴展名依舊使用的是“.sass”,這也就造成血案了,編譯時說編譯不出來。在此特別提醒新同學:“.sass”只能使用 Sass 老語法規則(縮進規則),“.scss”使用的是 Sass 的新語法規則,也就是 SCSS 語法規則(類似 CSS 語... ...
  • 主要需要註意的是不同方法他們本身返回的值應該是什麼,是數組當前的長度,還是取出的元素的值,再在splice函數裡面進行相應的return就可以了。具體如下: 用 splice函數實現 push方法 用 splice函數實現 pop方法 用 splice函數實現 shift方法 用 splice函數實 ...
  • 功能:停止事件冒泡 function stopBubble(e) { // 如果提供了事件對象,則這是一個非IE瀏覽器 if ( e && e.stopPropagation ) { // 因此它支持W3C的stopPropagation()方法 e.stopPropagation(); } els ...
  • 1、JavaScript對象的創建方式 在JavaScript中,創建對象的方式包括兩種:對象字面量和使用new表達式。對象字面量是一種靈活方便的書寫方式,例如: 1 2 3 4 5 6 var o1 = { p:”I’m in Object literal”, alertP:function(){ ...
  • 1、萬惡的回調 對前端工程師來說,非同步回調是再熟悉不過了,瀏覽器中的各種交互邏輯都是通過事件回調實現的,前端邏輯越來越複雜,導致回調函數越來越多,同時 nodejs 的流行也讓 javascript 在後端的複雜場景中得到應用,在 nodejs 代碼中更是經常看到層層嵌套。非同步操作的回調一旦嵌套很多 ...
  • ruby 標記定義ruby註釋(中文註音或字元)。ruby標記與rt標記一同使用。ruby標記由一個或多個字元(需要一個解釋/發音)和一個提供該信息的rt 標記組成,還包括可選的rp標記,定義當瀏覽器不支持ruby 標記時顯示的內容。 基本語法: 效果圖: 註意: 基本語法內rp標記與rt標記要註意 ...
  • guava是google的一個開源java框架,其github地址是 https://github.com/google/guava。guava工程包含了若幹被Google的 Java項目廣泛依賴的核心庫,例如:集合 [collections] 、緩存 [caching] 、原生類型支持 [prim ...
  • 電商支付架構設計交易核心支付編排------------------------------------------------------------------今天先到這兒,希望對您技術領導力, 企業管理,系統架構設計與評估,團隊管理, 項目管理, 產品管理,團隊建設 有參考作用 , 您可能感興... ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...