從C++單例模式到線程安全詳解
先看一個(gè)最簡(jiǎn)單的教科書(shū)式單例模式:
class CSingleton
{
public:
static CSingleton* getInstance()
{
if (NULL == ps)
{//tag1
ps = new CSingleton;
}
return ps;
}
private:
CSingleton(){}
CSingleton & operator=(const CSingleton &s);
static CSingleton* ps;
};
CSingleton* CSingleton::ps = NULL;
有2個(gè)要點(diǎn):
1.private的構(gòu)造函數(shù)和=操作符,用于防止類外的實(shí)例化和被復(fù)制;
2.static的類指針和get方法。
在大多數(shù)單線程情況下,以上代碼大都會(huì)運(yùn)行得很好,除非遇到中斷:
1.當(dāng)程序運(yùn)行到tag1 處觸發(fā)了中斷;
2.中斷處理程序恰調(diào)用的也是getInstance函數(shù)。
可想而知,這和多線程的情況類似,假設(shè)線程A 運(yùn)行到tag1處,還沒(méi)來(lái)得及new,此時(shí)ps仍然是NULL,線程B(或中斷處理程序) 同時(shí)也運(yùn)行到此通過(guò)if判斷,那么將會(huì)實(shí)例化2個(gè)CSingleton對(duì)象,顯然是不對(duì)的。
為了解決上述問(wèn)題,自然而然,最容易想到也最常用的方法是加鎖,因此getInstance改成這樣:
static CSingleton* getInstance()
{
lock();//偽代碼
if (NULL == ps)
{
ps = new CSingleton;
}
return ps;
}
加了鎖以后貌似解決了上述問(wèn)題,但也同樣帶來(lái)了新的問(wèn)題:如果程序到處是諸如:
CSingleton::instance()->aaaa(); CSingleton::instance()->bbbb(); CSingleton::instance()->cccc();
這樣的調(diào)用,除了第一次的lock()有用外,后面的都是在做無(wú)用功,lock()的代價(jià)說(shuō)大不大,但在某些情況下還是會(huì)提高程序延遲,這對(duì)追求完美的程序猿來(lái)說(shuō)是完全無(wú)法接受的。
于是乎,咱想出了一個(gè)辦法:
static CSingleton* getInstance()
{
if (NULL == ps)//這里加了次判斷,只有第一次才會(huì)為true而調(diào)用lock()
{
lock();//偽代碼
if (NULL == ps)
{
ps = new CSingleton;
}
}
return ps;
}
很久以后我才知道,這個(gè)方法有個(gè)很高大上的名字,叫做雙重檢查鎖定模式,簡(jiǎn)稱DCLP(Double Checked Locking Pattern)。
DCLP很好地解決了多次調(diào)用不必要的lock()。
然而,你們以為這樣就完了?too young。。
DCLP在多線程下仍然存在2個(gè)根本問(wèn)題:
1.程序的指令執(zhí)行順序不確定;
2.編譯器優(yōu)化問(wèn)題。
先說(shuō)2,在某些編譯器下,以上的兩個(gè)if判斷只會(huì)執(zhí)行一個(gè),甚至一個(gè)都不執(zhí)行,原因是編譯器認(rèn)為至少有一個(gè)if判斷是多余的,它自動(dòng)幫助我們優(yōu)化了代碼。
再說(shuō)1,ps = new CSingleton; 這條語(yǔ)句會(huì)被拆分為這樣的三個(gè)步驟執(zhí)行:
1.為要new的對(duì)象開(kāi)辟一塊內(nèi)存;
2.構(gòu)造該對(duì)象,填入這塊內(nèi)存;
3.將ps指針指向這塊內(nèi)存。
以上三個(gè)步驟,2和3的順序是不確定的,可能先2后3,也可能先3后2。。。
實(shí)際執(zhí)行時(shí)可能是這樣的:
static CSingleton* getInstance()
{
if (NULL == ps)
{
lock();//偽代碼
if (NULL == ps)
{ //偽代碼
ps = xx;//step 3
new sizeof(CSingleton);//step 1
new CSingleton;//step 2
}
}
return ps;
}
如果編譯器按上述順序執(zhí)行代碼,考慮如下?tīng)顩r:
線程A 執(zhí)行到step 1還未執(zhí)行后面的step 2,此時(shí)ps非空,但其指向的內(nèi)存里面的內(nèi)容還未被構(gòu)造出來(lái),于此同時(shí)線程B 進(jìn)入這個(gè)函數(shù),判斷ps非空直接返回ps,但是調(diào)用者此時(shí)訪問(wèn)的ps內(nèi)存實(shí)際內(nèi)容CSingleton還沒(méi)被構(gòu)造呢,這是一塊地址正確大小正確但內(nèi)部數(shù)據(jù)不明的東西,當(dāng)然會(huì)出錯(cuò)(調(diào)用者一般這么調(diào)用:CSingleton::getInstance()->aa(); CSingleton::getInstance()->bb(); CSingleton::getInstance()->cc();........此時(shí)的aa,bb,cc是啥玩意兒?)。
這也是為什么加上volatile關(guān)鍵字仍然不可以解決同步問(wèn)題,volatile只解決了編譯器優(yōu)化問(wèn)題,卻無(wú)法控制機(jī)器指令執(zhí)行順序。
很遺憾的是,C/C++本身在設(shè)計(jì)時(shí)是不考慮多線程問(wèn)題的,也就是說(shuō),要處理多線程問(wèn)題還要程序猿自己想辦法填坑。。
說(shuō)了這么多,我們要討論的問(wèn)題仍然沒(méi)有解決,慶幸的是,C++ 11提供了內(nèi)存柵欄技術(shù)來(lái)解決這個(gè)問(wèn)題,這里不贅述,有興趣的讀者可以自己搜索資料看看,不過(guò)是一些api調(diào)罷了。
那么,C++ 11 以前的代碼如何解決這個(gè)問(wèn)題呢?很不幸,并沒(méi)有很好的解決方案,一種可行的方案是,程序中不要到處這么調(diào)用這個(gè)單例對(duì)象:
CSingleton::getInstance()->aa(); CSingleton::getInstance()->bb(); CSingleton::getInstance()->cc();
而是在程序開(kāi)始就初始化緩存這個(gè)單例對(duì)象:
CSingleton* const g_ps = CSingleton::getInstance();//程序一開(kāi)始就緩存這個(gè)單例對(duì)象 g_ps->aa(); g_ps->bb(); g_ps->cc();
但是如此帶來(lái)的問(wèn)題是程序一開(kāi)始就實(shí)例化了這個(gè)單例對(duì)象,對(duì)象在整個(gè)程序的聲明周期存在,這貌似叫餓漢式,而之前那種叫懶漢式,孰輕孰重,只有根據(jù)實(shí)際情況取舍了。
以上就是小編為大家?guī)?lái)的從C++單例模式到線程安全詳解全部?jī)?nèi)容了,希望大家多多支持我們~
欄 目:C語(yǔ)言
下一篇:C++時(shí)間戳轉(zhuǎn)換成日期時(shí)間的步驟和示例代碼
本文標(biāo)題:從C++單例模式到線程安全詳解
本文地址:http://www.jygsgssxh.com/a1/Cyuyan/1941.html
您可能感興趣的文章
- 04-02c語(yǔ)言沒(méi)有round函數(shù) round c語(yǔ)言
- 01-10深入理解C++中常見(jiàn)的關(guān)鍵字含義
- 01-10使用C++實(shí)現(xiàn)全排列算法的方法詳解
- 01-10c++中inline的用法分析
- 01-10用C++實(shí)現(xiàn)DBSCAN聚類算法
- 01-10全排列算法的非遞歸實(shí)現(xiàn)與遞歸實(shí)現(xiàn)的方法(C++)
- 01-10C++大數(shù)模板(推薦)
- 01-10淺談C/C++中的static與extern關(guān)鍵字的使用詳解
- 01-10深入C/C++浮點(diǎn)數(shù)在內(nèi)存中的存儲(chǔ)方式詳解
- 01-10深入理解C/C++混合編程


閱讀排行
- 1C語(yǔ)言 while語(yǔ)句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹(shù)的示例代碼(圣誕
- 3利用C語(yǔ)言實(shí)現(xiàn)“百馬百擔(dān)”問(wèn)題方法
- 4C語(yǔ)言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語(yǔ)言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語(yǔ)言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語(yǔ)言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 04-02c語(yǔ)言函數(shù)調(diào)用后清空內(nèi)存 c語(yǔ)言調(diào)用
- 04-02func函數(shù)+在C語(yǔ)言 func函數(shù)在c語(yǔ)言中
- 04-02c語(yǔ)言的正則匹配函數(shù) c語(yǔ)言正則表達(dá)
- 04-02c語(yǔ)言用函數(shù)寫(xiě)分段 用c語(yǔ)言表示分段
- 04-02c語(yǔ)言中對(duì)數(shù)函數(shù)的表達(dá)式 c語(yǔ)言中對(duì)
- 04-02c語(yǔ)言編寫(xiě)函數(shù)冒泡排序 c語(yǔ)言冒泡排
- 04-02c語(yǔ)言沒(méi)有round函數(shù) round c語(yǔ)言
- 04-02c語(yǔ)言分段函數(shù)怎么求 用c語(yǔ)言求分段
- 04-02C語(yǔ)言中怎么打出三角函數(shù) c語(yǔ)言中怎
- 04-02c語(yǔ)言調(diào)用函數(shù)求fibo C語(yǔ)言調(diào)用函數(shù)求
隨機(jī)閱讀
- 01-11ajax實(shí)現(xiàn)頁(yè)面的局部加載
- 01-10使用C語(yǔ)言求解撲克牌的順子及n個(gè)骰子
- 01-11Mac OSX 打開(kāi)原生自帶讀寫(xiě)NTFS功能(圖文
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 04-02jquery與jsp,用jquery
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 01-10SublimeText編譯C開(kāi)發(fā)環(huán)境設(shè)置
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 01-10C#中split用法實(shí)例總結(jié)
- 01-10delphi制作wav文件的方法


