GJM : C#設計模式(1)——單例模式

来源:http://www.cnblogs.com/GJM6/archive/2016/11/27/6106718.html
-Advertisement-
Play Games

感謝您的閱讀。喜歡的、有用的就請大哥大嫂們高抬貴手“推薦一下”吧!你的精神支持是博主強大的寫作動力以及轉載收藏動力。歡迎轉載! 版權聲明:本文原創發表於 【請點擊連接前往】 ,未經作者同意必須保留此段聲明!如有問題請聯繫我,侵立刪,謝謝! 我的博客:http://www.cnblogs.com/GJ ...


  • 感謝您的閱讀。喜歡的、有用的就請大哥大嫂們高抬貴手“推薦一下”吧!你的精神支持是博主強大的寫作動力以及轉載收藏動力。歡迎轉載!
  • 版權聲明:本文原創發表於 【請點擊連接前往】 ,未經作者同意必須保留此段聲明!如有問題請聯繫我,侵立刪,謝謝!
  • 我的博客:http://www.cnblogs.com/GJM6/  -  傳送門:【點擊前往

 

一、引言

     主要的參考書籍是《Head First 設計模式》。首先我介紹的是設計模式中比較簡單的一個模式——單例模式(因為這裡只牽涉到一個類)

二、單例模式的介紹

     單例模式 : 從字面理解為一個類只有一個示例

                   所以單例模式也就是保證一個類只有一個實例的一種實現方法罷了

                   (設計模式其實就是幫助我們解決實際開發過程中的方法, 該方法是為了降低對象之間的耦合度,然而解決方法有很多種,所以前人就總結了一些常用的解決方法為書籍,從而把這本書就稱為設計模式)

      下麵給出單例模式的一個官方定義:確保一個類只有一個實例,並提供一個全局訪問點。

      為了幫助大家更好地理解單例模式,大家可以結合下麵的類圖來進行理解,以及後面也會剖析單例模式的實現思路:

三、為什麼會有單例模式

           看完單例模式的介紹,自然大家都會有這樣一個疑問——為什麼要有單例模式的?它在什麼情況下使用的?從單例模式的定義中我們可以看出——單例模式的使用自然是當我們的系統中某個對象只需要一個實例的情況,例如:操作系統中只能有一個任務管理器,操作文件時,同一時間內只允許一個實例對其操作等,既然現實生活中有這樣的應用場景,自然在軟體設計領域必須有這樣的解決方案了(因為軟體設計也是現實生活中的抽象),所以也就有了單例模式了。

四、剖析單例模式的實現思路

瞭解完了一些關於單例模式的基本概念之後,下麵就為大家剖析單例模式的實現思路的,因為在我自己學習單例模式的時候,咋一看單例模式的實現代碼確實很簡單,也很容易看懂,但是我還是覺得它很陌生(這個可能是看的少的,或者自己在寫代碼中也用的少的緣故),而且心裡總會這樣一個疑問——為什麼前人會這樣去實現單例模式的呢?他們是如何思考的呢?後面經過自己的琢磨也就慢慢理清楚單例模式的實現思路了,並且此時也不再覺得單例模式模式的,下麵就分享我的一個剖析過程的:

我們從單例模式的概念(確保一個類只有一個實例,並提供一個訪問它的全局訪問點)入手,可以把概念進行拆分為兩部分:(1)確保一個類只有一個實例;(2)提供一個訪問它的全局訪問點;下麵通過採用兩人對話的方式來幫助大家更快掌握分析思路:

菜鳥:怎樣確保一個類只有一個實例了?

老鳥:那就讓我幫你分析下,你創建類的實例會想到用什麼方式來創建的呢?

新手:用new關鍵字啊,只要new下就創建了該類的一個實例了,之後就可以使用該類的一些屬性和實例方法了

老鳥:那你想過為什麼可以使用new關鍵字來創建類的實例嗎?

菜鳥:這個還有條件的嗎?………, 哦,我想起來了,如果類定義私有的構造函數就不能在外界通過new創建實例了(註:有些初學者就會問,有時候我並沒有在類中定義構造函數為什麼也可以使用new來創建對象,那是因為編譯器在背後做了手腳了,當編譯器看到我們類中沒有定義構造函數,此時編譯器會幫我們生成一個公有的無參構造函數)

老鳥:不錯,回答的很對,這樣你的疑惑就得到解答了啊

菜鳥:那我要在哪裡創建類的實例了?

老鳥:你傻啊,當然是在類裡面創建了(註:這樣定義私有構造函數就是上面的一個思考過程的,要創建實例,自然就要有一個變數來保存該實例把,所以就有了私有變數的聲明,但是實現中是定義靜態私有變數,朋友們有沒有想過——這裡為什麼定義為靜態的呢?對於這個疑問的解釋為:每個線程都有自己的線程棧,定義為靜態主要是為了在多線程確保類有一個實例

菜鳥:哦,現在完全明白了,但是我還有另一個疑問——現在類實例創建在類內部,那外界如何獲得該的一個實例來使用它了?

老鳥:這個,你可以定義一個公有方法或者屬性來把該類的實例公開出去了(註:這樣就有了公有方法的定義了,該方法就是提供方法問類的全局訪問點)

通過上面的分析,相信大家也就很容易寫出單例模式的實現代碼了,下麵就看看具體的實現代碼(看完之後你會驚訝道:真是這樣的!):

          C#  
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 /// <summary>     /// 單例模式的實現     /// </summary>     public class Singleton     {         // 定義一個靜態變數來保存類的實例         private static Singleton uniqueInstance;           // 定義私有構造函數,使外界不能創建該類實例         private Singleton()         {         }           /// <summary>         /// 定義公有方法提供一個全局訪問點,同時你也可以定義公有屬性來提供全局訪問點         /// </summary>         /// <returns></returns>         public static Singleton GetInstance()         {             // 如果類的實例不存在則創建,否則直接返回             if (uniqueInstance == null)             {                 uniqueInstance = new Singleton();             }             return uniqueInstance;         }     }

上面的單例模式的實現在單線程下確實是完美的,然而在多線程的情況下會得到多個Singleton實例,因為在兩個線程同時運行GetInstance方法時,此時兩個線程判斷(uniqueInstance ==null)這個條件時都返回真,此時兩個線程就都會創建Singleton的實例,這樣就違背了我們單例模式初衷了,既然上面的實現會運行多個線程執行,那我們對於多線程的解決方案自然就是使GetInstance方法在同一時間只運行一個線程運行就好了,也就是我們線程同步的問題了,具體的解決多線程的代碼如下:

            C#  
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 /// <summary>     /// 單例模式的實現     /// </summary>     public class Singleton     {         // 定義一個靜態變數來保存類的實例         private static Singleton uniqueInstance;           // 定義一個標識確保線程同步         private static readonly object locker = new object();           // 定義私有構造函數,使外界不能創建該類實例         private Singleton()         {         }           /// <summary>         /// 定義公有方法提供一個全局訪問點,同時你也可以定義公有屬性來提供全局訪問點         /// </summary>         /// <returns></returns>         public static Singleton GetInstance()         {             // 當第一個線程運行到這裡時,此時會對locker對象 "加鎖",             // 當第二個線程運行該方法時,首先檢測到locker對象為"加鎖"狀態,該線程就會掛起等待第一個線程解鎖             // lock語句運行完之後(即線程運行完之後)會對該對象"解鎖"             lock (locker)             {                 // 如果類的實例不存在則創建,否則直接返回                 if (uniqueInstance == null)                 {                     uniqueInstance = new Singleton();                 }             }               return uniqueInstance;         }     }

上面這種解決方案確實可以解決多線程的問題,但是上面代碼對於每個線程都會對線程輔助對象locker加鎖之後再判斷實例是否存在,對於這個操作完全沒有必要的,因為當第一個線程創建了該類的實例之後,後面的線程此時只需要直接判斷(uniqueInstance==null)為假,此時完全沒必要對線程輔助對象加鎖之後再去判斷,所以上面的實現方式增加了額外的開銷,損失了性能,為了改進上面實現方式的缺陷,我們只需要在lock語句前面加一句(uniqueInstance==null)的判斷就可以避免鎖所增加的額外開銷,這種實現方式我們就叫它 “雙重鎖定”,下麵具體看看實現代碼的:

            C#  
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 /// <summary>     /// 單例模式的實現     /// </summary>     public class Singleton     {         // 定義一個靜態變數來保存類的實例         private static Singleton uniqueInstance;           // 定義一個標識確保線程同步         private static readonly object locker = new object();           // 定義私有構造函數,使外界不能創建該類實例         private Singleton()         {         }           /// <summary>         /// 定義公有方法提供一個全局訪問點,同時你也可以定義公有屬性來提供全局訪問點         /// </summary>         /// <returns></returns>         public static Singleton GetInstance()         {             // 當第一個線程運行到這裡時,此時會對locker對象 "加鎖",             // 當第二個線程運行該方法時,首先檢測到locker對象為"加鎖"狀態,該線程就會掛起等待第一個線程解鎖             // lock語句運行完之後(即線程運行完之後)會對該對象"解鎖"             // 雙重鎖定只需要一句判斷就可以了             if (uniqueInstance == null)             {                 lock (locker)                 {                     // 如果類的實例不存在則創建,否則直接返回                     if (uniqueInstance == null)                     {                         uniqueInstance = new Singleton();                     }                 }             }             return uniqueInstance;         }     }

 

五、C#中實現了單例模式的類

理解完了單例模式之後,菜鳥又接著問了:.NET FrameWork類庫中有沒有單例模式的實現呢?

經過查看,.NET類庫中確實存在單例模式的實現類,不過該類不是公開的,下麵就具體看看該類的一個實現的(該類具體存在於System.dll程式集,命名空間為System,大家可以用反射工具Reflector去查看源碼的):

          C#  
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 // 該類不是一個公開類     // 但是該類的實現應用了單例模式     internal sealed class SR     {         private static SR loader;         internal SR()         {         }         // 主要是因為該類不是公有,所以這個全部訪問點也定義為私有的了         // 但是思想還是用到了單例模式的思想的         private static SR GetLoader()         {             if (loader == null)             {                 SR sr = new SR();                 Interlocked.CompareExchange<SR>(ref loader, sr, null);             }             return loader;         }           // 這個公有方法中調用了GetLoader方法的         public static object GetObject(string name)         {             SR loader = GetLoader();             if (loader == null)             {                 return null;             }             return loader.resources.GetObject(name, Culture);         }     }

 

 

六、總結

到這裡,設計模式的單例模式就介紹完了,希望通過本文章大家可以對單例模式有一個更深的理解,並且希望之前沒接觸過單例模式或覺得單例模式陌生的朋友看完之後會驚嘆:原來如此!


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

-Advertisement-
Play Games
更多相關文章
  • 一、抽象工廠模式 在工廠方法中, 所創建的汽車都是沒有品牌的汽車. 現在假設, 車子有品牌了, 並且不止只有汽車, 還有自行車的生產. 那麼可以把汽車和自行車做成兩個抽象產品, 代碼如下: 為什麼要把產品做成抽象的呢, 因為雖然各品牌的產品大體相同, 但是其中很多細節還是不一樣的. 所以就產生了兩種 ...
  • 上一篇, 介紹了簡單工廠模式, 在最後提出了一個問題, 如果我的程式需要擴展, 加一種產品進去, 顯然, 簡單工廠是不能勝任此項工作的, 那麼需要另請高明 - 工廠方法模式. 當然, 抽象工廠, 也是可以加產品的, 動態添加程式集, 然後反射的方式就可以創建出我們新加的對象. 不過此處, 主要還是介 ...
  • 一、Servlet簡介 Servlet是sun公司提供的一門用於開發動態web資源的技術。 Sun公司在其API中提供了一個servlet介面,用戶若想用發一個動態web資源(即開發一個Java程式向瀏覽器輸出數據),需要完成以下2個步驟: 1、編寫一個Java類,實現servlet介面。 2、把開 ...
  • 前言: "軟體模式的偉大之處, 就在於他們傳達了許多有用的設計思想. 所以, 在學習了大量模式之後, 就理應成為非常優秀的軟體設計人員, 不是嗎? 當學習、使用了幾十個設計模式之後, 我也曾這樣認為. 模式幫助我開發靈活的框架, 幫助我構建堅固、可擴展的軟體系統. 但是幾年之後, 我卻發現自己在模式 ...
  • 每天一個設計模式-7 生成器模式(Builder) 一、實際問題 在討論工廠方法模式的時候,提到了一個導出數據的應用框架,但是並沒有涉及到導出數據的具體實現,這次通過生成器模式來簡單實現導出成文本,Xml等具體的格式。 導出成文本或Xml等格式的數據時,一般都會有各自的格式,比如:導出的文件都有3個 ...
  • 內容摘要從架構的本質,軟體工程,架構師職責,成長路徑等方面,介紹什麼是架構,架構流程以及架構師職責和成長規劃。 本篇主題一、什麼是架構二、項目中的角色三、架構師職責和工作內容四、架構工作流五、架構師成長路徑六、架構能力模型七、擴展知識八、本章總結 一、什麼是架構架構是系統的藍圖,描述了系統的結構和關 ...
  • 1.json 2.HotFragment ...
  • 1.json 2.按照指定比例展示寬高的自定義控制項實現 為了讓圖片按照完美比例進行展現, 不被壓縮, 需要自定義控制項,該控制項可以根據預設的比例來確定寬高 自定義屬性 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...