中文字幕免费精品_亚洲视频自拍_亚洲综合国产激情另类一区_色综合咪咪久久

asp.net中C++單例實現問題分析
來源:易賢網 閱讀:778 次 日期:2014-10-16 10:40:15
溫馨提示:易賢網小編為您整理了“asp.net中C++單例實現問題分析”,方便廣大網友查閱!

方案一

代碼如下:

class QMManager

{

public:

static QMManager &instance()

{

static QMManager instance_;

return instance_;

}

}

這是最簡單的版本,在單線程下(或者是C++0X下)是沒任何問題的,但在多線程下就不行了,因為static QMManager instance_;這句話不是線程安全的。

在局部作用域下的靜態變量在編譯時,編譯器會創建一個附加變量標識靜態變量是否被初始化,會被編譯器變成像下面這樣(偽代碼):

代碼如下:

static QMManager &instance()

{

static bool constructed = false;

static uninitialized QMManager instance_;

if (!constructed) {

constructed = true;

new(&s) QMManager; //construct it

}

return instance_;

}

這里有競爭條件,兩個線程同時調用instance()時,一個線程運行到if語句進入后還沒設constructed值,此時切換到另一線程,constructed值還是false,同樣進入到if語句里初始化變量,兩個線程都執行了這個單例類的初始化,就不再是單例了。

方案二

一個解決方法是加鎖:

代碼如下:

static QMManager &instance()

{

Lock(); //鎖自己實現

static QMManager instance_;

UnLock();

return instance_;

}

但這樣每次調用instance()都要加鎖解鎖,代價略大。

方案三

那再改變一下,把內部靜態實例變成類的靜態成員,在外部初始化,也就是在include了文件,main函數執行前就初始化這個實例,就不會有線程重入問題了:

代碼如下:

class QMManager

{

protected:

static QMManager instance_;

QMManager();

~QMManager(){};

public:

static QMManager *instance()

{

return &instance_;

}

void do_something();

};

QMManager QMManager::instance_; //外部初始化

這被稱為餓漢模式,程序一加載就初始化,不管有沒有調用到。

看似沒問題,但還是有坑,在一個2B情況下會有問題:在這個單例類的構造函數里調用另一個單例類的方法可能會有問題。

看例子:

代碼如下:

//.h

class QMManager

{

protected:

static QMManager instance_;

QMManager();

~QMManager(){};

public:

static QMManager *instance()

{

return &instance_;

}

};

class QMSqlite

{

protected:

static QMSqlite instance_;

QMSqlite();

~QMSqlite(){};

public:

static QMSqlite *instance()

{

return &instance_;

}

void do_something();

};

QMManager QMManager::instance_;

QMSqlite QMSqlite::instance_;

//.cpp

QMManager::QMManager()

{

printf("QMManager constructorn");

QMSqlite::instance()->do_something();

}

QMSqlite::QMSqlite()

{

printf("QMSqlite constructorn");

}

void QMSqlite::do_something()

{

printf("QMSqlite do_somethingn");

}

這里QMManager的構造函數調用了QMSqlite的instance函數,但此時QMSqlite::instance_可能還沒有初始化。

這里的執行流程:程序開始后,在執行main前,執行到QMManager QMManager::instance_;這句代碼,初始化QMManager里的instance_靜態變量,調用到QMManager的構造函數,在構造函數里調用QMSqlite::instance(),取QMSqlite里的instance_靜態變量,但此時QMSqlite::instance_還沒初始化,問題就出現了。

那這里會crash嗎,測試結果是不會,這應該跟編譯器有關,靜態數據區空間應該是先被分配了,在調用QMManager構造函數前,QMSqlite成員函數在內存里已經存在了,只是還未調到它的構造函數,所以輸出是這樣:

QMManager constructor

QMSqlite do_something

QMSqlite constructor

方案四

那這個問題怎么解決呢,單例對象作為靜態局部變量有線程安全問題,作為類靜態全局變量在一開始初始化,有以上2B問題,那結合下上述兩種方式,可以解決這兩個問題。boost的實現方式是:單例對象作為靜態局部變量,但增加一個輔助類讓單例對象可以在一開始就初始化。如下:

代碼如下:

//.h

class QMManager

{

protected:

struct object_creator

{

object_creator()

{

QMManager::instance();

}

inline void do_nothing() const {}

};

static object_creator create_object_;

QMManager();

~QMManager(){};

public:

static QMManager *instance()

{

static QMManager instance;

return &instance;

}

};

QMManager::object_creator QMManager::create_object_;

class QMSqlite

{

protected:

QMSqlite();

~QMSqlite(){};

struct object_creator

{

object_creator()

{

QMSqlite::instance();

}

inline void do_nothing() const {}

};

static object_creator create_object_;

public:

static QMSqlite *instance()

{

static QMSqlite instance;

return &instance;

}

void do_something();

};

QMManager::object_creator QMManager::create_object_;

QMSqlite::object_creator QMSqlite::create_object_;

結合方案3的.cpp,這下可以看到正確的輸出和調用了:

QMManager constructor

QMSqlite constructor

QMSqlite do_something

來看看這里的執行流程:

初始化QMManager類全局靜態變量create_object_

->調用object_creator的構造函數

->調用QMManager::instance()方法初始化單例

->執行QMManager的構造函數

->調用QMSqlite::instance()

->初始化局部靜態變量QMSqlite instance

->執行QMSqlite的構造函數,然后返回這個單例。

跟方案三的區別在于QMManager調用QMSqlite單例時,方案3是取到全局靜態變量,此時這個變量未初始化,而方案四的單例是靜態局部變量,此時調用會初始化。

跟最初方案一的區別是在main函數前就初始化了單例,不會有線程安全問題。

最終boost

上面為了說明清楚點去除了模版,實際使用是用模版,不用寫那么多重復代碼,這是boost庫的模板實現:

代碼如下:

template <typename T>

struct Singleton

{

struct object_creator

{

object_creator(){ Singleton<T>::instance(); }

inline void do_nothing()const {}

};

static object_creator create_object;

public:

typedef T object_type;

static object_type& instance()

{

static object_type obj;

//據說這個do_nothing是確保create_object構造函數被調用

//這跟模板的編譯有關

create_object.do_nothing();

return obj;

}

};

template <typename T> typename Singleton<T>::object_creator Singleton<T>::create_object;

class QMManager

{

protected:

QMManager();

~QMManager(){};

friend class Singleton<QMManager>;

public:

void do_something(){};

};

int main()

{

Singleton<QMManager>::instance()->do_something();

return 0;

}

其實Boost庫這樣的實現像打了幾個補丁,用了一些奇技淫巧,雖然確實繞過了坑實現了需求,但感覺挺不好的。

更多信息請查看IT技術專欄

更多信息請查看網絡編程
易賢網手機網站地址:asp.net中C++單例實現問題分析
由于各方面情況的不斷調整與變化,易賢網提供的所有考試信息和咨詢回復僅供參考,敬請考生以權威部門公布的正式信息和咨詢為準!

2026上岸·考公考編培訓報班

  • 報班類型
  • 姓名
  • 手機號
  • 驗證碼
關于我們 | 聯系我們 | 人才招聘 | 網站聲明 | 網站幫助 | 非正式的簡要咨詢 | 簡要咨詢須知 | 新媒體/短視頻平臺 | 手機站點 | 投訴建議
工業和信息化部備案號:滇ICP備2023014141號-1 云南省教育廳備案號:云教ICP備0901021 滇公網安備53010202001879號 人力資源服務許可證:(云)人服證字(2023)第0102001523號
云南網警備案專用圖標
聯系電話:0871-65099533/13759567129 獲取招聘考試信息及咨詢關注公眾號:hfpxwx
咨詢QQ:1093837350(9:00—18:00)版權所有:易賢網
云南網警報警專用圖標
中文字幕免费精品_亚洲视频自拍_亚洲综合国产激情另类一区_色综合咪咪久久
亚洲欧美第一页| 欧美日韩综合视频| 欧美性大战xxxxx久久久| 亚洲欧美日本国产专区一区| 亚洲第一成人在线| 国产综合18久久久久久| 日韩五码在线| 日韩一级免费观看| 在线一区免费观看| 一区二区三区精品| 在线中文字幕不卡| 亚洲免费影视第一页| 亚洲午夜精品久久| 18成人免费观看视频| 国产日韩一区二区三区| 狠狠色狠狠色综合日日91app| 亚洲国产精品高清久久久| 亚洲国产激情| 日韩亚洲在线观看| 午夜视频一区| 久久成人一区| 欧美日韩一区综合| 国产精品国内视频| 狠狠爱成人网| 亚洲一区在线视频| 亚洲视频一区| 欧美精品久久久久久久久老牛影院| 亚洲男人的天堂在线| 欧美一区免费视频| 久久久高清一区二区三区| 免费成人在线观看视频| 欧美性猛交xxxx乱大交蜜桃| 黄色一区二区在线| 亚洲一区二区三区中文字幕| 久久夜色精品国产欧美乱极品| 欧美国产日本| 韩国女主播一区| 中文欧美字幕免费| 免费亚洲婷婷| 国产精品国产一区二区 | 欧美精品v国产精品v日韩精品| 免费欧美网站| 国产欧美日韩另类视频免费观看| 亚洲欧洲精品一区二区三区不卡 | 久久亚洲高清| 国产精品青草久久| 免费在线亚洲| 激情偷拍久久| 国产伦精品一区二区三区免费| 久久久久久久久岛国免费| 欧美日韩天天操| 在线观看亚洲视频| 久久精品日产第一区二区三区| 欧美日韩美女在线| 久久青青草综合| 亚洲国产欧美一区| 亚洲国产日韩精品| 好看的亚洲午夜视频在线| 欧美日韩一区成人| 国产精品高清网站| 欧美护士18xxxxhd| 欧美日韩午夜激情| 欧美精品一区二区三区蜜桃| 久久一区二区视频| 久久亚洲春色中文字幕久久久 | 影音先锋国产精品| 伊人久久亚洲美女图片| 亚洲国产二区| 亚洲欧洲午夜| 亚洲欧美日韩国产另类专区| 欧美在线视频免费观看| 欧美一区国产二区| 久久精品国产亚洲精品| 欧美精品成人一区二区在线观看| 欧美三级电影大全| 国产精品免费区二区三区观看| 免费在线一区二区| 欧美精品福利| 亚洲国产成人精品久久久国产成人一区| 91久久国产自产拍夜夜嗨| 久久亚洲综合| 在线电影院国产精品| 久久蜜桃资源一区二区老牛| 国产一区二区三区在线观看免费视频 | 亚洲精品少妇网址| 欧美一区二区三区啪啪| 欧美猛交免费看| 国产综合久久| 欧美影院精品一区| 国产综合精品| 久久在线视频在线| 亚洲国产欧美不卡在线观看 | 一区二区三区.www| 男人天堂欧美日韩| 日韩视频免费观看高清在线视频| 欧美 日韩 国产在线| 亚洲激情在线观看视频免费| 欧美激情 亚洲a∨综合| 日韩视频专区| 激情久久中文字幕| 久久综合色影院| 一区二区欧美视频| 国产一区二区中文| 欧美激情国产日韩精品一区18| 一本色道久久综合亚洲精品按摩| 国产精品毛片a∨一区二区三区|国 | 国产精品久久久久久久久免费桃花| 亚洲手机成人高清视频| 国产精品美女久久久久久久| 亚洲一区在线看| 亚洲黄色精品| 国产日韩精品一区观看| 蜜臀久久99精品久久久久久9| 午夜精品久久一牛影视| 亚洲日本乱码在线观看| 国产女主播一区二区| 欧美了一区在线观看| 欧美一区二区精美| 亚洲午夜国产成人av电影男同| 在线看视频不卡| 国产区欧美区日韩区| 欧美四级在线观看| 欧美激情自拍| 女人色偷偷aa久久天堂| 欧美与欧洲交xxxx免费观看 | 国产日韩欧美黄色| 国产精品婷婷| 欧美午夜精品理论片a级按摩| 蜜桃视频一区| 欧美国产亚洲视频| 欧美激情2020午夜免费观看| 老司机免费视频一区二区| 久久免费视频在线| 噜噜噜91成人网| 欧美成人情趣视频| 欧美成人一品| 亚洲欧美日韩区| 欧美一级欧美一级在线播放| 性色一区二区| 亚洲美女在线视频| 亚洲激情专区| 国产精品99久久久久久白浆小说| 一本一本久久| 亚洲欧美在线高清| 久久精品国产99| 男男成人高潮片免费网站| 欧美日韩国产在线播放| 欧美视频在线观看免费网址| 欧美三级资源在线| 国产亚洲激情在线| 亚洲精品女av网站| 欧美在线999| 欧美日韩一区二区三区四区在线观看| 欧美日韩免费观看一区| 国产亚洲a∨片在线观看| 亚洲精品看片| 午夜日韩在线观看| 久久久久免费视频| 欧美性理论片在线观看片免费| 国产一区二区高清| 在线亚洲+欧美+日本专区| 美女精品一区| 国产婷婷色一区二区三区四区| 亚洲精品日韩久久| 另类专区欧美制服同性| 国产午夜精品久久| 亚洲欧美成人| 欧美午夜精品久久久久久孕妇| 亚洲激情电影在线| 久久久久天天天天| 在线观看欧美视频| 美女福利精品视频| 亚洲第一福利社区| 免费看黄裸体一级大秀欧美| 国产偷自视频区视频一区二区| 亚洲日本欧美在线| 欧美区一区二区三区| 亚洲人妖在线| 欧美日韩色一区| 中文精品在线| 国产美女精品视频| 久久精品国亚洲| 影音先锋中文字幕一区二区| 久久精品国产亚洲5555| 狠狠久久亚洲欧美专区| 久久久久久久网站| 亚洲欧洲精品成人久久奇米网| 欧美成人精品激情在线观看| 91久久线看在观草草青青| 欧美理论电影网| 亚洲欧美日韩精品在线| 国产亚洲综合性久久久影院| 久久人人爽人人| 在线日韩成人| 欧美另类99xxxxx| 午夜视频久久久久久| 国内成人在线| 国产精品久久久久久超碰| 久久久久久亚洲精品中文字幕 | 亚洲一区不卡|