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

asp.net中C++單例實現問題分析
來源:易賢網 閱讀:807 次 日期:2014-11-01 08:21:28
溫馨提示:易賢網小編為您整理了“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)版權所有:易賢網
云南網警報警專用圖標
中文字幕免费精品_亚洲视频自拍_亚洲综合国产激情另类一区_色综合咪咪久久
久久精品国产亚洲精品 | 中国成人在线视频| 亚洲国产老妈| 亚洲清纯自拍| 亚洲一区二区在线观看视频| 久久se精品一区精品二区| 免费成人高清| 国产精品制服诱惑| 国产一区二区三区黄视频| 黄色成人在线观看| 亚洲视频在线看| 欧美成人在线影院| 国产在线高清精品| 中国av一区| 欧美乱妇高清无乱码| 国产一区二区三区免费观看| 9l国产精品久久久久麻豆| 久久久另类综合| 国产日韩精品一区二区浪潮av| 日韩一级欧洲| 欧美伦理在线观看| 亚洲国产一区在线| 蜜臀av性久久久久蜜臀aⅴ| 国产亚洲一区在线| 欧美一区二区三区免费看| 国产精品卡一卡二| 亚洲一级在线观看| 国产精品大片wwwwww| av成人免费观看| 欧美日本一道本在线视频| 亚洲国产日韩欧美| 欧美国产第二页| 最新亚洲电影| 欧美激情一区在线观看| 亚洲精品日韩综合观看成人91| 噜噜噜在线观看免费视频日韩| 国产一区二区视频在线观看| 亚久久调教视频| 国产中文一区二区| 美国成人直播| 日韩午夜激情av| 欧美亚洲成人网| 午夜精品一区二区三区在线| 国产精品久久久一区二区| 亚洲一二三区在线| 国产欧美日韩视频一区二区| 欧美在线免费视频| 黑人巨大精品欧美一区二区小视频| 久久久999| 亚洲精品欧美在线| 国产精品国产三级国产专播品爱网 | 国产精品久久久久久久久搜平片| 亚洲美女在线观看| 欧美激情综合在线| 一本色道久久综合精品竹菊 | 最新成人av在线| 麻豆精品在线播放| 亚洲黄色影片| 欧美性大战xxxxx久久久| 亚洲综合大片69999| 国产精品久久综合| 久久九九热免费视频| 亚洲国产一区二区精品专区| 欧美日韩国产一区二区三区| 亚洲人体影院| 欧美日韩精品一区二区在线播放| 亚洲精品国产精品乱码不99按摩| 欧美视频一区二区三区在线观看 | 欧美日韩亚洲一区二区三区在线观看| 99这里只有精品| 国产精品久久久久一区二区三区| 久久久久亚洲综合| 中国女人久久久| 国产日韩欧美自拍| 美女诱惑一区| 亚洲自拍偷拍麻豆| 亚洲国产日韩欧美在线图片| 欧美裸体一区二区三区| 亚洲一区二区三区中文字幕| 韩日精品视频| 欧美日韩在线大尺度| 美女精品网站| 亚洲欧美综合另类中字| 亚洲经典三级| 国产日韩精品视频一区| 欧美日韩免费观看一区| 久久久久女教师免费一区| 中文网丁香综合网| 亚洲大片在线观看| 国产欧美在线播放| 欧美日韩视频在线| 欧美成人a视频| 欧美在线观看日本一区| 亚洲狼人综合| 揄拍成人国产精品视频| 国产精品网站在线| 欧美日韩日日夜夜| 麻豆精品在线观看| 欧美亚洲三区| 欧美一区三区三区高中清蜜桃| 国产乱人伦精品一区二区| 免费观看成人www动漫视频| 亚洲欧美国产高清| 一本久久综合| 亚洲理论电影网| 在线日韩中文字幕| 精东粉嫩av免费一区二区三区| 国产精品欧美风情| 欧美日韩亚洲一区二| 欧美11—12娇小xxxx| 久久婷婷av| 久久精视频免费在线久久完整在线看| 在线视频你懂得一区二区三区| 亚洲黄色精品| 国精产品99永久一区一区| 国产精品美女久久久免费| 欧美极品aⅴ影院| 欧美第一黄色网| 免费视频一区二区三区在线观看| 久久九九精品| 久久久精品动漫| 欧美一区二区三区啪啪| 中国成人在线视频| 亚洲女性喷水在线观看一区| 亚洲欧美卡通另类91av| 在线一区二区视频| 亚洲视频免费| 亚洲欧美一区二区视频| 在线一区二区视频| 亚洲欧美一区二区激情| 久久久精品日韩| 久久久噜噜噜| 蜜桃av一区二区| 欧美精品在线观看播放| 欧美日韩中文字幕精品| 欧美日韩调教| 欧美精品色一区二区三区| 欧美日韩在线综合| 国产亚洲网站| 亚洲激情一区| 亚洲一区二区久久| 久久久久久久97| 欧美区日韩区| 国产在线成人| 在线视频一区观看| 欧美在线国产| 欧美日韩hd| 狠狠色丁香婷婷综合影院| 亚洲精品国精品久久99热| 亚洲欧美日韩在线播放| 久久夜色精品国产欧美乱极品| 欧美久久视频| 精品成人乱色一区二区| 亚洲男人的天堂在线| 免费观看成人www动漫视频| 国产精品毛片va一区二区三区 | 欧美中文字幕精品| 欧美华人在线视频| 国产欧美亚洲日本| 99在线观看免费视频精品观看| 欧美主播一区二区三区美女 久久精品人| 久久久亚洲高清| 国产精品国产三级国产 | 蜜桃av久久久亚洲精品| 国产麻豆日韩欧美久久| 悠悠资源网亚洲青| 亚洲一区二区免费看| 美女露胸一区二区三区| 欧美三级日本三级少妇99| 一区免费视频| 久久精品国产69国产精品亚洲| 欧美福利视频网站| 国产在线精品成人一区二区三区 | 欧美人成网站| 在线日韩成人| 午夜精品一区二区在线观看| 蜜月aⅴ免费一区二区三区| 国产丝袜一区二区| 亚洲性图久久| 欧美视频在线观看免费网址| 国内外成人免费视频| 亚洲一区二区三区涩| 欧美精品一区二区视频| 最新亚洲一区| 欧美福利视频在线| 亚洲欧洲在线免费| 久久九九免费| 在线观看日韩av先锋影音电影院| 欧美一区二区三区精品电影| 国产精品国产亚洲精品看不卡15| 亚洲电影免费在线观看| 欧美黄色成人网| 亚洲精品国产视频| 另类欧美日韩国产在线| 国内精品免费午夜毛片| 欧美伊人精品成人久久综合97| 国产欧美不卡| 久久夜色精品国产欧美乱| 一区二区在线观看av| 免费久久精品视频|