淺談單例模式_第1頁
淺談單例模式_第2頁
淺談單例模式_第3頁
淺談單例模式_第4頁
淺談單例模式_第5頁
已閱讀5頁,還剩3頁未讀 繼續(xù)免費閱讀

下載本文檔

版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進行舉報或認領(lǐng)

文檔簡介

前言什么是單例模式?單例模式,屬于創(chuàng)建類型的一種常用的軟件設(shè)計模式。通過單例模式的方法創(chuàng)建的類在當前進程中只有一個實例(根據(jù)需要,也有可能一個線程中屬于單例,如:僅線程上下文內(nèi)使用同一個實例)上面是百度百科給出的解釋。大家都知道,面向?qū)ο蟮乃枷刖褪俏覀兛梢园岩粋€類實例很多次,每次實例出來的都是一個對象,意味著你可以創(chuàng)建很多個基于這個類的對象。單例模式,說白了,就是這些對象本質(zhì)都是同一個,整個程序中,不管在哪里用,使用的都是同一個實例對象。如果我們創(chuàng)建了一個China類,我們可以一直new嗎?不可以,因為世界上只有一個China,所以我們使用的都是同一個China對象。Version1-非線程安全

publicclassChina

{

private

China()

{

}

privatestaticChinachina

=

null;

publicstaticChinaInstance

{

get

{

if

(china

==

null)

{

Console.WriteLine("實例化對象");

china

=

newChina();

}

return

china;

}

}

}最簡單的實現(xiàn)方式如上圖,創(chuàng)建一個私有的靜態(tài)對象和私有構(gòu)造方法,然后在CreateInstance方法里,加一個判斷,如果為Null,就重新實例化一下,否則直接返回。這種寫法從邏輯上是沒問題的,但是是否會出現(xiàn)這個if(china==null)判斷,同時執(zhí)行,這樣就麻煩了。所以這種寫法在單線程的程序是沒問題的,但是在多線程中,是可能會有問題的。我們做個測試,測試代碼如下:

classProgram

{

staticvoidMain(string[]

args)

{

for

(inti

=

0;

i

<

10;

i++)

{

newTaskFactory().StartNew(()

=>

{

Chinachina

=

China.Instance;

});

Thread.Sleep(10);

}

Console.ReadLine();

}

}上面的代碼,就是創(chuàng)建10個線程,都執(zhí)行CreatInstance方法,那么最終是輸出多少次Console.WriteLine("實例化對象")呢?我們測試發(fā)現(xiàn),這個輸出結(jié)果是不唯一的,有時候會輸出5次,有時候會輸出2次,但是一般都是超過1次,這個就說明對象被多次實例化了,這就違背了單例模式的原則。Version2-簡單的線程安全既然出現(xiàn)問題,那么我們就需要做一下優(yōu)化,優(yōu)化之后的代碼如下:

publicclassChina

{

private

China()

{

}

privatestaticChinachina

=

null;

privatestaticobjectobjlock

=

newobject();

publicstaticChinaInstance

{

get

{

lock

(objlock)

{

Console.WriteLine("執(zhí)行判斷");

if

(china

==

null)

{

Console.WriteLine("實例化對象");

china

=

newChina();

}

}

return

china;

}

}

}對比看下,就是加了一個同步鎖,這樣就可以避免同時執(zhí)行的情況,并且,我們在lock里加了一個Console.WriteLine("執(zhí)行判斷"),觀察這行代碼執(zhí)行多少次。從結(jié)果來看,實例化對象只執(zhí)行了一次,說明對象只被創(chuàng)建過一次,滿足了我們的需求,達到了預期的效果。Version3-雙if+lock實現(xiàn)上面那種方式已經(jīng)可以達到預期效果,但是我們注意到一個問題,執(zhí)行判斷這行代碼被執(zhí)行了10次,這顯示不符合我們的邏輯,既然已經(jīng)實例化了,為什么每次還要執(zhí)行判斷呢?是不是多此一舉?并且每次請求對象,都會進行l(wèi)ock操作,lock對性能是有一定影響的。于是我們繼續(xù)優(yōu)化,優(yōu)化之后的代碼如下:

publicclassChina

{

private

China()

{

}

privatestaticChinachina

=

null;

privatestaticobjectobjlock

=

newobject();

publicstaticChinaInstance

{

get

{

if

(china

==

null)

{

lock

(objlock)

{

Console.WriteLine("執(zhí)行判斷");

if

(china

==

null)

{

Console.WriteLine("實例化對象");

china

=

newChina();

}

}

}

return

china;

}

}

}我們對比代碼可以看出,就是又加了一個if(china==null),這種雙if+lock的方式,是不是可以解決我們的問題呢?我們執(zhí)行一次,看看結(jié)果:我們通過結(jié)果可以看到只執(zhí)行了一次判斷,也只執(zhí)行一次實例化對象,但是我們還可以繼續(xù)優(yōu)化。Version4-靜態(tài)變量實現(xiàn)話不多說,直接上代碼:

publicclassChina

{

private

China()

{

}

privatestatic

readonly

Chinachina

=

newChina();

publicstaticChinaInstance

{

get

{

return

china;

}

}

}利用靜態(tài)變量去實現(xiàn)單例,非常簡單,但同時也是線程安全的,由CLR保證,在程序第一次使用該類之前被調(diào)用,而且只調(diào)用一次。但是這種方式也有缺點,就是實例化過程是在程序初始化時就執(zhí)行的,而不是在使用時才執(zhí)行,就是說,不管你用不用,都已經(jīng)實例化了。Version5-完全懶漢式實現(xiàn)

publicclassChina

{

private

China()

{

}

publicstaticChinaInstance

{

get

{

return

Lazy.instance;

}

}

privateclassLazy

{

static

Lazy()

{

}

internalstatic

readonly

Chinainstance

=

newChina();

}

}這種方法與上一種方法類似,只是多加了一個類,來解決上一個版本的缺點。Version6-使用Lazy特性從.NET4開始,可以使用Lazytype來實現(xiàn)完全懶漢式,代碼也變得更簡單,代碼如下:

publicclassChina

{

private

China()

{

}

privatestatic

readonly

Lazy<China>

lazy

=

newLazy<China>(()

=>

newChina());

publicstaticChinaInstance

{

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預覽,若沒有圖紙預覽就沒有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 人人文庫網(wǎng)僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負責。
  • 6. 下載文件中如有侵權(quán)或不適當內(nèi)容,請與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

評論

0/150

提交評論