C 設計模式 1 單例模式

2021-06-27 08:58:07 字數 3900 閱讀 7983

最近在設計模式的一些內容,主要的參考書籍是《head first 設計模式》,同時在學習過程中也檢視了很多中關於設計模式的一些文章的,在這裡記錄下我的一些學習筆記,一是為了幫助我更深入地理解設計模式,二同時可以給一些初學設計模式的朋友一些參考。首先我介紹的是設計模式中比較簡單的乙個模式——單例模式(因為這裡只牽涉到乙個類)

說到單例模式,大家第一反應應該就是——什麼是單例模式?,從「單例」字面意思上理解為——乙個類只有乙個例項,所以單例模式也就是保證乙個類只有乙個例項的一種實現方法罷了(設計模式其實就是幫助我們解決實際開發過程中的方法, 該方法是為了降低物件之間的耦合度,然而解決方法有很多種,所以前人就總結了一些常用的解決方法為書籍,從而把這本書就稱為設計模式),下面給出單例模式的乙個官方定義:確保乙個類只有乙個例項,並提供乙個全域性訪問點。為了幫助大家更好地理解單例模式,大家可以結合下面的類圖來進行理解,以及後面也會剖析單例模式的實現思路:

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

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

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

菜鳥:怎樣確保乙個類只有乙個例項了?

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

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

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

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

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

菜鳥:那我要在**建立類的例項了?

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

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

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

///

///單例模式的實現

/// public

class

singleton

//////

定義公有方法提供乙個全域性訪問點,同時你也可以定義公有屬性來提供全域性訪問點

/// ///

public

static

singleton getinstance()

return

uniqueinstance;}}

上面的單例模式的實現在單執行緒下確實是完美的,然而在多執行緒的情況下會得到多個singleton例項,因為在兩個執行緒同時執行getinstance方法時,此時兩個執行緒判斷(uniqueinstance ==null)這個條件時都返回真,此時兩個執行緒就都會建立singleton的例項,這樣就違背了我們單例模式初衷了,既然上面的實現會執行多個執行緒執行,那我們對於多執行緒的解決方案自然就是使getinstance方法在同一時間只執行乙個執行緒執行就好了,也就是我們執行緒同步的問題了(對於執行緒同步大家也可以參考我執行緒同步的文章),具體的解決多執行緒的**如下:

///

///單例模式的實現

/// public

class

singleton

//////

定義公有方法提供乙個全域性訪問點,同時你也可以定義公有屬性來提供全域性訪問點

/// ///

public

static

singleton getinstance()

}return

uniqueinstance;}}

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

///

///單例模式的實現

/// public

class

singleton

//////

定義公有方法提供乙個全域性訪問點,同時你也可以定義公有屬性來提供全域性訪問點

/// ///

public

static

singleton getinstance()}}

return

uniqueinstance;}}

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

經過檢視,.net類庫中確實存在單例模式的實現類,不過該類不是公開的,下面就具體看看該類的乙個實現的(該類具體存在於system.dll程式集,命名空間為system,大家可以用反射工具reflector去檢視原始碼的):

//

該類不是乙個公開類

//但是該類的實現應用了單例模式

internal

sealed

class

sr

//主要是因為該類不是公有,所以這個全部訪問點也定義為私有的了

//但是思想還是用到了單例模式的思想的

private

static

sr getloader()

return

loader;

}//這個公有方法中呼叫了getloader方法的

public

static

object getobject(string

name)

return

loader.resources.getobject(name, culture);}}

C 設計模式(1) 單例模式

單例模式的特點 1 該類只有乙個例項 2 該類在內部建立該例項 3 向整個系統公開這個例項介面 適用於多執行緒的四種程式設計方式 第一種 將建構函式設為私有可以保證其他類不能從這個類派生或者建立類的例項 另外 析構函式私有化則是為了讓某個類只能通過new來建立 class singletion1 p...

C 設計模式 1 單例模式

原文 c 設計模式 1 單例模式 最近在設計模式的一些內容,主要的參考書籍是 head first 設計模式 同時在學習過程中也檢視了很多中關於設計模式的一些文章的,在這裡記錄下我的一些學習筆記,一是為了幫助我更深入地理解設計模式,二同時可以給一些初學設計模式的朋友一些參考。首先我介紹的是設計模式中...

C 設計模式 1 單例模式

最近在設計模式的一些內容,主要的參考書籍是 head first 設計模式 同時在學習過程中也檢視了很多中關於設計模式的一些文章的,在這裡記錄下我的一些學習筆記,一是為了幫助我更深入地理解設計模式,二同時可以給一些初學設計模式的朋友一些參考。首先我介紹的是設計模式中比較簡單的乙個模式 單例模式 因為...