亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > C++ > 正文

從C++單例模式到線程安全詳解

2020-05-23 13:57:41
字體:
來源:轉載
供稿:網友

先看一個最簡單的教科書式單例模式:

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個要點:

1.private的構造函數和=操作符,用于防止類外的實例化和被復制;

2.static的類指針和get方法。

在大多數單線程情況下,以上代碼大都會運行得很好,除非遇到中斷:

1.當程序運行到tag1 處觸發了中斷;
2.中斷處理程序恰調用的也是getInstance函數。

可想而知,這和多線程的情況類似,假設線程A 運行到tag1處,還沒來得及new,此時ps仍然是NULL,線程B(或中斷處理程序) 同時也運行到此通過if判斷,那么將會實例化2個CSingleton對象,顯然是不對的。

為了解決上述問題,自然而然,最容易想到也最常用的方法是加鎖,因此getInstance改成這樣:

	static CSingleton* getInstance()	{		lock();//偽代碼		if (NULL == ps)		{			ps = new CSingleton;		}		return ps;	}

加了鎖以后貌似解決了上述問題,但也同樣帶來了新的問題:如果程序到處是諸如:

CSingleton::instance()->aaaa();CSingleton::instance()->bbbb();CSingleton::instance()->cccc();

這樣的調用,除了第一次的lock()有用外,后面的都是在做無用功,lock()的代價說大不大,但在某些情況下還是會提高程序延遲,這對追求完美的程序猿來說是完全無法接受的。

于是乎,咱想出了一個辦法:

	static CSingleton* getInstance()	{		if (NULL == ps)//這里加了次判斷,只有第一次才會為true而調用lock()		{			lock();//偽代碼			if (NULL == ps)			{				ps = new CSingleton;			}		}		return ps;	}

很久以后我才知道,這個方法有個很高大上的名字,叫做雙重檢查鎖定模式,簡稱DCLP(Double Checked Locking Pattern)。

DCLP很好地解決了多次調用不必要的lock()。

然而,你們以為這樣就完了?too young。。

DCLP在多線程下仍然存在2個根本問題:

1.程序的指令執行順序不確定;
2.編譯器優化問題。

先說2,在某些編譯器下,以上的兩個if判斷只會執行一個,甚至一個都不執行,原因是編譯器認為至少有一個if判斷是多余的,它自動幫助我們優化了代碼。

再說1,ps = new CSingleton; 這條語句會被拆分為這樣的三個步驟執行:

1.為要new的對象開辟一塊內存;
2.構造該對象,填入這塊內存;
3.將ps指針指向這塊內存。

以上三個步驟,2和3的順序是不確定的,可能先2后3,也可能先3后2。。。

實際執行時可能是這樣的:

	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;	}

如果編譯器按上述順序執行代碼,考慮如下狀況:

線程A 執行到step 1還未執行后面的step 2,此時ps非空,但其指向的內存里面的內容還未被構造出來,于此同時線程B 進入這個函數,判斷ps非空直接返回ps,但是調用者此時訪問的ps內存實際內容CSingleton還沒被構造呢,這是一塊地址正確大小正確但內部數據不明的東西,當然會出錯(調用者一般這么調用:CSingleton::getInstance()->aa();  CSingleton::getInstance()->bb();  CSingleton::getInstance()->cc();........此時的aa,bb,cc是啥玩意兒?)。

這也是為什么加上volatile關鍵字仍然不可以解決同步問題,volatile只解決了編譯器優化問題,卻無法控制機器指令執行順序。

很遺憾的是,C/C++本身在設計時是不考慮多線程問題的,也就是說,要處理多線程問題還要程序猿自己想辦法填坑。。

說了這么多,我們要討論的問題仍然沒有解決,慶幸的是,C++ 11提供了內存柵欄技術來解決這個問題,這里不贅述,有興趣的讀者可以自己搜索資料看看,不過是一些api調罷了。

那么,C++ 11 以前的代碼如何解決這個問題呢?很不幸,并沒有很好的解決方案,一種可行的方案是,程序中不要到處這么調用這個單例對象:

CSingleton::getInstance()->aa(); CSingleton::getInstance()->bb();CSingleton::getInstance()->cc();

而是在程序開始就初始化緩存這個單例對象:

CSingleton* const g_ps = CSingleton::getInstance();//程序一開始就緩存這個單例對象g_ps->aa();g_ps->bb();g_ps->cc();

但是如此帶來的問題是程序一開始就實例化了這個單例對象,對象在整個程序的聲明周期存在,這貌似叫餓漢式,而之前那種叫懶漢式,孰輕孰重,只有根據實際情況取舍了。

以上就是小編為大家帶來的從C++單例模式到線程安全詳解全部內容了,希望大家多多支持VEVB武林網~


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
在线播放亚洲激情| 精品成人69xx.xyz| 日韩精品中文字幕在线播放| 992tv在线成人免费观看| 亚洲精品视频中文字幕| 一本色道久久88亚洲综合88| 国产成人一区二区| 欧美激情亚洲国产| 国产精品99一区| 91视频免费在线| 久久人人爽亚洲精品天堂| 日本不卡免费高清视频| 97超级碰在线看视频免费在线看| 国产欧美日韩中文字幕| 亚洲最大福利视频| 日韩成人在线观看| 欧美日韩综合视频网址| 久久偷看各类女兵18女厕嘘嘘| 韩国视频理论视频久久| 日韩在线观看av| 97婷婷涩涩精品一区| 国产日韩精品在线播放| 欧美日韩在线视频一区| 国产97在线观看| 日韩欧美在线视频免费观看| 亚洲精品有码在线| 国产精品video| 欧美日韩中文字幕日韩欧美| 日韩中文字幕不卡视频| 久久中国妇女中文字幕| 日韩免费看的电影电视剧大全| 国产欧美精品在线| 亚洲欧美成人精品| www高清在线视频日韩欧美| 亚洲成人三级在线| 欧美在线中文字幕| 91青草视频久久| 国产狼人综合免费视频| 日韩在线不卡视频| 久久精品视频免费播放| 欧美日韩午夜激情| 国产在线高清精品| 欧美日韩中文字幕在线视频| 中文字幕日韩高清| 国产一区二区在线免费| 欧美高清无遮挡| 欧美大成色www永久网站婷| 欧美精品xxx| 91久久久久久久久| 粗暴蹂躏中文一区二区三区| 97福利一区二区| 一区二区三区四区在线观看视频| 中文字幕精品—区二区| 日韩欧美有码在线| 国内精品美女av在线播放| 欧美国产第一页| 欧美又大粗又爽又黄大片视频| 日韩免费高清在线观看| 成人性生交xxxxx网站| 久久香蕉国产线看观看av| 久久精品久久久久久国产 免费| 久久综合伊人77777蜜臀| 九九热这里只有精品免费看| 韩国精品美女www爽爽爽视频| 久久国内精品一国内精品| 久久久欧美一区二区| 国产在线拍揄自揄视频不卡99| 国产一区二区丝袜高跟鞋图片| 国产噜噜噜噜噜久久久久久久久| 精品视频—区二区三区免费| 欧美激情高清视频| 欧美成人精品h版在线观看| 亚洲男人7777| 日韩欧美中文免费| 欧美成人一区二区三区电影| 亚洲九九九在线观看| 欧美高跟鞋交xxxxhd| 亚洲精品在线看| 久久久久久久999精品视频| 亚洲激情中文字幕| 欧美一二三视频| 国产精品青青在线观看爽香蕉| 狠狠爱在线视频一区| 538国产精品一区二区在线| 国产精品www| 欧美性做爰毛片| 久久成年人视频| 亚洲欧美日韩一区二区三区在线| 黑人巨大精品欧美一区二区免费| 亚洲精品国产综合久久| 久久久成人的性感天堂| 亚洲国产欧美久久| 国产亚洲精品一区二区| 国产精品美女免费| 久久成年人视频| 亚洲已满18点击进入在线看片| 麻豆成人在线看| 亚洲无亚洲人成网站77777| www.亚洲人.com| 亚洲全黄一级网站| 色偷偷9999www| 成人免费直播live| 亚洲综合成人婷婷小说| 亚洲国产精品99久久| 国产女人18毛片水18精品| 亚洲欧美制服中文字幕| 97成人精品视频在线观看| 乱亲女秽乱长久久久| 国产一区二区三区三区在线观看| 欧美日韩国产一区二区| 欧美与黑人午夜性猛交久久久| 国产精品久久av| 欧美激情视频免费观看| 亚洲精品电影网| 欧美壮男野外gaytube| 亚洲国产中文字幕在线观看| 亚洲乱亚洲乱妇无码| 欧美自拍视频在线| 亚洲a区在线视频| 亚洲人成网7777777国产| 欧美性少妇18aaaa视频| 国产香蕉一区二区三区在线视频| 91日韩在线播放| 成人精品久久一区二区三区| 在线观看久久av| 久久久久久成人| 成人激情电影一区二区| 九九久久国产精品| 亚洲精品资源美女情侣酒店| 欧美性生交xxxxxdddd| 国产成人精品久久二区二区91| 国产一区二区丝袜| 91探花福利精品国产自产在线| 久久久人成影片一区二区三区| 日韩精品在线观看视频| 97成人超碰免| 欧美精品videofree1080p| 97精品国产97久久久久久春色| 欧美主播福利视频| 91精品久久久久久久久不口人| 国产精品www| 日韩精品中文字幕有码专区| 亚洲人成网7777777国产| 日本国产欧美一区二区三区| 亚洲性日韩精品一区二区| 国产综合香蕉五月婷在线| 亚洲影视九九影院在线观看| 最近的2019中文字幕免费一页| 日韩av中文在线| 国产精品永久免费| 黄色成人在线播放| 在线观看91久久久久久| 日韩在线观看你懂的| 欧美日韩国产精品一区| 日韩成人网免费视频| 精品国内自产拍在线观看| 亚洲第一中文字幕在线观看| 中文字幕亚洲字幕| 欧美日韩午夜视频在线观看| 久久久久久69| 亚洲精品日产aⅴ| 91久久精品国产91久久| 欧美日本黄视频| 一区二区三区视频免费在线观看|