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

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

使用設計模式中的單例模式來實現C++的boost庫

2020-05-23 14:08:30
字體:
來源:轉載
供稿:網友
這篇文章主要介紹了使用設計模式中的單例模式來實現C++的boost庫的方法,其中作者對線程安全格外強調,需要的朋友可以參考下
 

線程安全的單例模式
一、懶漢模式
:即第一次調用該類實例的時候才產生一個新的該類實例,并在以后僅返回此實例。

需要用鎖,來保證其線程安全性:原因:多個線程可能進入判斷是否已經存在實例的if語句,從而non thread safety。

使用double-check來保證thread safety。但是如果處理大量數據時,該鎖才成為嚴重的性能瓶頸。

1、靜態成員實例的懶漢模式:

class Singleton{private:  static Singleton* m_instance;  Singleton(){}public:  static Singleton* getInstance();}; Singleton* Singleton::getInstance(){  if(NULL == m_instance)  {    Lock();//借用其它類來實現,如boost    if(NULL == m_instance)    {      m_instance = new Singleton;    }    UnLock();  }  return m_instance;}

2、內部靜態實例的懶漢模式

這里需要注意的是,C++0X以后,要求編譯器保證內部靜態變量的線程安全性,可以不加鎖。但C++ 0X以前,仍需要加鎖。

class SingletonInside{private:  SingletonInside(){}public:  static SingletonInside* getInstance()  {    Lock(); // not needed after C++0x    static SingletonInside instance;    UnLock(); // not needed after C++0x    return instance;   }};

二、餓漢模式:即無論是否調用該類的實例,在程序開始時就會產生一個該類的實例,并在以后僅返回此實例。

由靜態初始化實例保證其線程安全性,WHY?因為靜態實例初始化在程序開始時進入主函數之前就由主線程以單線程方式完成了初始化,不必擔心多線程問題。

故在性能需求較高時,應使用這種模式,避免頻繁的鎖爭奪。

class SingletonStatic{private:  static const SingletonStatic* m_instance;  SingletonStatic(){}public:  static const SingletonStatic* getInstance()  {    return m_instance;  }}; //外部初始化 before invoke mainconst SingletonStatic* SingletonStatic::m_instance = new SingletonStatic;


boost庫的實現示例

單例本來是個很簡單的模式,實現上應該也是很簡單,但C++單例的簡單實現會有一些坑,有了上面線程安全的基礎,下面來看看為了避免這些坑怎樣一步步演化到boost庫的實現方式。

方案一

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情況下會有問題:在這個單例類的構造函數里調用另一個單例類的方法可能會有問題。
看例子:

//.hclass 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_;//.cppQMManager::QMManager(){  printf("QMManager constructor/n");  QMSqlite::instance()->do_something();} QMSqlite::QMSqlite(){  printf("QMSqlite constructor/n");}void QMSqlite::do_something(){  printf("QMSqlite do_something/n");}

這里QMManager的構造函數調用了QMSqlite的instance函數,但此時QMSqlite::instance_可能還沒有初始化。
這里的執行流程:程序開始后,在執行main前,執行到QMManager QMManager::instance_;這句代碼,初始化QMManager里的instance_靜態變量,調用到QMManager的構造函數,在構造函數里調用QMSqlite::instance(),取QMSqlite里的instance_靜態變量,但此時QMSqlite::instance_還沒初始化,問題就出現了。
那這里會crash嗎,測試結果是不會,這應該跟編譯器有關,靜態數據區空間應該是先被分配了,在調用QMManager構造函數前,QMSqlite成員函數在內存里已經存在了,只是還未調到它的構造函數,所以輸出是這樣:

QMManager constructorQMSqlite do_somethingQMSqlite constructor

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

//.hclass 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 constructorQMSqlite constructorQMSqlite 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庫這樣的實現像打了幾個補丁,用了一些奇技淫巧,雖然確實繞過了坑實現了需求,但感覺挺不好的。



發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲japanese制服美女| 色偷偷888欧美精品久久久| 久久亚洲精品中文字幕冲田杏梨| 欧美在线一区二区视频| 亚洲视屏在线播放| 色综合伊人色综合网| 亚洲成色www8888| 57pao成人国产永久免费| 亚洲精品黄网在线观看| 在线精品视频视频中文字幕| 在线播放日韩av| 中文字幕亚洲精品| 国产成人一区二区在线| 中文字幕亚洲一区二区三区五十路| 久久久精品一区二区| 亚洲色在线视频| 亚洲福利视频在线| 日韩有码视频在线| 欧美激情视频一区二区| 日韩av影视综合网| 国产精品视频资源| 国产香蕉精品视频一区二区三区| 在线亚洲国产精品网| 日本欧美精品在线| 欧美日韩国产丝袜另类| 亚洲国产高清高潮精品美女| 国产精品久久久久久久电影| 亚洲一品av免费观看| 欧美另类极品videosbestfree| 国产福利精品视频| 在线观看久久久久久| 国产精品一区二区久久久久| 最好看的2019年中文视频| 亚洲人成网站免费播放| 日韩人体视频一二区| 中文字幕亚洲二区| 国产99在线|中文| 日韩大陆毛片av| 欧美激情一区二区三区在线视频观看| 精品国产拍在线观看| 久久天天躁狠狠躁夜夜爽蜜月| 亚洲品质视频自拍网| 国产精品99久久久久久白浆小说| 17婷婷久久www| 久久久久久久亚洲精品| 欧美日韩国产成人高清视频| 国产99久久久欧美黑人| 97avcom| 国产精品第三页| 欧美日韩视频免费播放| www.久久色.com| 亚洲精品一区二区在线| 亚洲欧美一区二区三区情侣bbw| 国内精品久久久久久影视8| 国产成人一区二区三区小说| 久久频这里精品99香蕉| 亚洲国产另类久久精品| 91在线色戒在线| 亚洲少妇激情视频| 国产成人精品在线视频| 国产日韩欧美91| 欧美国产日韩一区二区| 欧美尤物巨大精品爽| 久久久在线观看| 国产成人精品免高潮在线观看| 欧美极度另类性三渗透| 亚洲成人亚洲激情| 亚洲第一在线视频| 欧美精品在线第一页| 亚洲综合av影视| 欧美黑人一区二区三区| 97视频在线播放| 久久精品免费播放| 668精品在线视频| 亚洲色图美腿丝袜| 国产精品69av| 中文.日本.精品| 91成人在线观看国产| 欧美精品电影在线| 亚洲国产精品va| 欧美精品18videosex性欧美| 亚洲乱亚洲乱妇无码| 欧美精品生活片| 国产精品激情av电影在线观看| 国产精品18久久久久久首页狼| 欧美激情精品久久久久久久变态| 在线精品国产欧美| 久久影院免费观看| 日韩高清电影好看的电视剧电影| 亚洲欧美制服综合另类| 日韩欧美在线视频| 亚洲色图美腿丝袜| 日韩av网址在线观看| 26uuu另类亚洲欧美日本老年| 欧美亚洲国产精品| 国产亚洲精品美女久久久久| 九九热视频这里只有精品| 欧美大全免费观看电视剧大泉洋| 久久人人97超碰精品888| 精品久久久久久中文字幕一区奶水| 欧美日韩精品国产| 欧美精品videossex88| 国产精品一区专区欧美日韩| 欧美日韩在线一区| 亚洲综合一区二区不卡| 国产免费一区二区三区在线观看| 精品久久久999| 亚洲午夜精品视频| 亚洲第一区在线观看| 亚洲xxxx妇黄裸体| 国产丝袜高跟一区| 日韩有码片在线观看| 精品亚洲夜色av98在线观看| 国产精品网站入口| 久久久久国产精品免费网站| 亚洲成人黄色网址| 久久久久久久久久久国产| 欧美性极品xxxx做受| 777午夜精品福利在线观看| 亚洲一级免费视频| 2019最新中文字幕| 精品国产一区二区三区久久久狼| 日韩国产在线播放| 欧美大胆在线视频| 日韩成人免费视频| 欧美性理论片在线观看片免费| 成人免费视频在线观看超级碰| 精品久久久久久久久久国产| 97碰在线观看| 亚洲精品国产综合区久久久久久久| 欧美视频一二三| 欧美—级高清免费播放| 欧美中文在线免费| 国产精品精品视频一区二区三区| 久久夜精品va视频免费观看| 国产精品网红福利| 精品视频中文字幕| 久久亚洲私人国产精品va| 日产精品99久久久久久| 欧美电影电视剧在线观看| 国产精品成av人在线视午夜片| 91亚洲精品在线观看| 亚洲a在线观看| 国产精品三级久久久久久电影| 国产精品精品一区二区三区午夜版| 国产一区二区三区日韩欧美| 色偷偷噜噜噜亚洲男人的天堂| 18一19gay欧美视频网站| 国产偷国产偷亚洲清高网站| 日韩中文字幕精品视频| 中文字幕日韩欧美在线| 欧美亚洲国产视频小说| 欧美激情精品久久久久久大尺度| 在线观看精品国产视频| 日韩国产精品亚洲а∨天堂免| 国产精品www网站| 亚洲国产成人一区| 日韩中文在线中文网在线观看| 中文字幕一精品亚洲无线一区| 国产精品一区专区欧美日韩| 黄色成人av在线| 午夜精品福利在线观看| 欧美激情一区二区三区久久久| 97国产精品人人爽人人做|