亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频

蟲蟲首頁| 資源下載| 資源專輯| 精品軟件
登錄| 注冊

您現(xiàn)在的位置是:首頁 > 技術(shù)閱讀 >  C++的單例模式為什么不直接全部使用static,而是非要實例化一個對象?

C++的單例模式為什么不直接全部使用static,而是非要實例化一個對象?

時間:2024-02-09

開場

前段時間我在知乎回答了這樣一個問題:

為什么C++單例模式不能直接全部使用 static變量和 static函數(shù)呢?如果全部使用 static的話,是不是也不會有多線程的問題了?而且“類型::方法”的訪問方式比起先getInstance()再訪問難道不是更加簡單清晰嗎?

(還是說是為了附和 “單例” 這樣一個字面上的意思)

//大概這個樣子
class Singleton {
public:
 static void on() {Singleton::isOn = true;}
 static void off() {Singleton::isOn = false;}
 static bool state() {return Singleton::isOn;}
private:
 static bool isOn;
};

這可能是很多C++學(xué)習(xí)者都會有的疑惑,下面是我的回答。

正文

通過getInstance()函數(shù)獲取單例對象,這種模式的關(guān)鍵之處不是在于強(qiáng)迫你用函數(shù)來獲取對象。關(guān)鍵之處是讓static對象定義在函數(shù)內(nèi)部,變成局部static變量。看下這種實現(xiàn)方式的經(jīng)典demo:

class Singleton {
public:
    static Singleton& getInstance() {
        static Singleton inst;
        return inst;
    }
    Singleton(const Singleton&) = delete;
    Singleton& operator=(const Singleton&) = delete;

    // 其他數(shù)據(jù)函數(shù)
    // ...

private:
    Singleton() { ... }
    // 其他數(shù)據(jù)成員
    // ...
};

學(xué)名是:Meyers' Singleton。沒錯,也就是說這是Scott Meyers最早提出來的C++單例模式的推薦寫法。

《Effective C++》系列叢書作者

???? 注意這種單例寫法需要C++11。因為是從C++11標(biāo)準(zhǔn)才開始規(guī)定 static變量是線程安全的。也就是說無需我們自己寫加鎖保護(hù)的代碼,編譯器能夠幫我們做到。

? 所以C++程序員們不要在讀完Java單例模式的資料之后,在C++程序中寫double check或volatile了!

如果是把 static對象定義成 Singleton的私有static成員變量,然后getInstance()去返回這個成員即:

class Singleton {
public:
    static Singleton& getInstance() {
        return inst;
    }
    Singleton(const Singleton&) = delete;
    Singleton& operator=(const Singleton&) = delete;

    // 其他數(shù)據(jù)函數(shù)
    // ...

private:
    Singleton() { ... }
    static Singleton inst;
    // 其他數(shù)據(jù)成員
    // ...
};
Singleton Singleton::inst;

雖然它也是 先getInstance()再訪問,但這種不是Meyers' Singleton


那么為什么Meyers推薦的是第一種的呢?

原因是這解決了一類重要問題,那就是static變量的初始化順序的問題。

C++只能保證在同一個文件中聲明的static變量的初始化順序與其變量聲明的順序一致。但是不能保證不同的文件中的static變量的初始化順序。

然后對于單例模式而言,不同的單例對象之間進(jìn)行調(diào)用也是常見的場景。比如我有一個單例,存儲了程序啟動時加載的配置文件的內(nèi)容。另外有一個單例,掌管著一個全局唯一的日志管理器。在日志管理初始化的時候,要通過配置文件的單例對象來獲取到某個配置項,實現(xiàn)日志打印。

這時候兩個單例在不同文件中各自實現(xiàn),很有可能在日志管理器的單例使用配置文件單例的時候,配置文件的單例對象是沒有被初始化的。這個未初始化可能產(chǎn)生的風(fēng)險指的是C++變量的未初始化,而不是說配置文件未加載的之類業(yè)務(wù)邏輯上的未初始化導(dǎo)致的問題。

Meyers' Singleton寫法中,單例對象是第一次訪問的時候(也就是第一次調(diào)用getInstance()函數(shù)的時候)才初始化的,但也是恰恰因為如此,因而能保證如果沒有初始化,在該函數(shù)調(diào)用的時候,是能完成初始化的。所以先getInstance()再訪問 這種形式的單例 其關(guān)鍵并不是在于這個形式。而是在于其內(nèi)容,局部static變量能保證通過函數(shù)來獲取static變量的時候,該函數(shù)返回的對象是肯定完成了初始化的!

講到這,我們對Meyers' Singleton的盲目鼓吹也需冷靜一下,因為C++同樣能保證所有文件內(nèi)(非函數(shù)內(nèi))的static變量在main()函數(shù)開始運(yùn)行之后肯定是都能做完初始化的。所以如果你是在main()函數(shù)運(yùn)行之后,用日志管理器的單例訪問配置文件的單例,那么其實也是沒有問題的… 這就引出Meyers' Singleton的第二個優(yōu)勢,那就是當(dāng)產(chǎn)生繼承的時候。如果出現(xiàn)繼承,這種寫法中:

class Singleton {
public:
    static void on() {Singleton::isOn = true;}
    static void off() {Singleton::isOn = false;}
    static bool state() {return Singleton::isOn;}
private:
    static bool isOn;
};

class Monitor: public Singleton {
public:
    static void addBrightness(int val) { brightness += val;}
    static void subBrightness(int val) { brightness -= val;}
    static int getBrightness() return brightness;}

private:
    static int brightness;
};

如果有子類繼承這一父類,來拓展成新的子類,比如Monitor顯示器類有開關(guān)狀態(tài),同時擴(kuò)展了一個亮度的成員。但是父子類的static成員變量是共享的,其isOn成員會有問題。

好吧,如果你說你的單例完全不會出現(xiàn)繼承的情況,是不是就不需要寫成Meyers' Singleton?我只想說,如果你一定要強(qiáng)加這么多限定的話,那么這種設(shè)計模式的討論本身就沒有意義。就很像是在說:我自己能夠保證每個new出來的指針我都能delete掉它,所以我不需要RAII……

???? 所謂設(shè)計模式(design pattern)、慣用法(idiom)這種老程序員的經(jīng)驗之談都是讓你在大多數(shù)情況下,即使你不懂其奧秘,但凡遵守了,就能避免掉很多潛在的問題。盡管這種問題并不能百分百發(fā)生。所以這倒沒必要去抬杠。

亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
在线电影国产精品| 国产主播一区二区三区四区| 亚洲免费精品| 欧美乱在线观看| 久久久水蜜桃| 国产在线精品一区二区中文| 美女图片一区二区| 亚洲韩国精品一区| 欧美理论大片| 午夜激情综合网| 国产综合色在线| 国产精品v欧美精品v日本精品动漫 | 蜜臀av一级做a爰片久久| 日韩天堂av| 国产精品午夜国产小视频| 久久精品人人爽| 亚洲精品久久久久久下一站| 国产婷婷成人久久av免费高清| 欧美成人免费播放| 久久爱www久久做| 亚洲欧美综合一区| 在线视频一区二区| 一区二区欧美日韩视频| 欧美新色视频| 国产精品视频内| 国产麻豆视频精品| 久久亚洲综合| 久久人91精品久久久久久不卡| 亚洲一区免费网站| 一区二区三区高清在线| 伊人久久噜噜噜躁狠狠躁| 国产日韩欧美日韩大片| 欧美视频二区36p| 欧美日韩视频在线第一区| 亚洲欧美日韩高清| 国内一区二区在线视频观看| 巨胸喷奶水www久久久免费动漫| 国产欧美日韩综合一区在线播放| 欧美伦理视频网站| 久久久久国产一区二区| 亚洲一区二区三区免费在线观看| 国外视频精品毛片| 欧美日韩一区二区在线播放| 久久综合一区二区| 午夜伦理片一区| 亚洲自拍偷拍福利| 国产日本欧美一区二区三区在线| 欧美激情一区二区在线 | 亚洲小说区图片区| 亚洲国产一区在线| 亚洲精品黄色| 久久久久高清| 麻豆国产va免费精品高清在线| 亚洲午夜激情在线| 亚洲欧美视频在线观看视频| 久久一二三四| 狠狠色狠狠色综合| 在线观看一区欧美| 亚洲欧美电影在线观看| 欧美96在线丨欧| 国产精品久久波多野结衣| 久久精品国产精品亚洲| 99精品国产在热久久婷婷| 亚洲精品国产精品国自产观看浪潮 | 最新国产精品拍自在线播放| 亚洲国产日日夜夜| 亚洲毛片在线看| 亚洲狠狠婷婷| 亚洲资源在线观看| 久久久精品tv| 欧美三级日本三级少妇99| 亚洲香蕉成视频在线观看| 欧美一区二区三区在线观看视频 | 国产精品毛片a∨一区二区三区| 国产精品尤物| 99国内精品| 久久免费精品视频| 欧美日韩精品是欧美日韩精品| 韩国三级在线一区| 亚洲欧美中文另类| 欧美人与性动交cc0o| 精品51国产黑色丝袜高跟鞋| 亚洲无毛电影| 欧美成人自拍| 欧美午夜精品久久久久久久| 亚洲国产精品va| 欧美成人精品高清在线播放| 亚洲大片在线| 欧美婷婷在线| 亚洲欧美在线网| 国产亚洲一区二区三区在线观看 | 欧美精品成人在线| 国产在线观看91精品一区| 亚洲一区二区三区四区在线观看| 性8sex亚洲区入口| 国产色产综合色产在线视频| 欧美在线视频一区| 亚洲福利视频网| 欧美精品久久一区二区| 亚洲视频一区在线| 国产视频欧美视频| 久久男女视频| 一本色道久久综合亚洲精品不 | 亚洲无线视频| 欧美午夜在线视频| 99国产精品私拍| 欧美激情a∨在线视频播放| 日韩视频永久免费| 国产视频精品va久久久久久| 久久久久久久精| 久久精品亚洲一区二区| 国产亚洲精品aa午夜观看| 免费日韩av片| 亚洲国产成人一区| 国产精品视频最多的网站| 久久一二三区| 欧美一区三区二区在线观看| 在线看国产一区| 国产日韩三区| 欧美精品手机在线| 欧美日韩精品综合| 国产一区二区三区的电影| 99精品视频免费全部在线| 国产免费观看久久| 国产精品久久久久久久app| 欧美国产欧美亚洲国产日韩mv天天看完整| 亚洲在线免费视频| 亚洲一区bb| 午夜精品婷婷| 亚洲欧美日本国产专区一区| 在线午夜精品自拍| 1024欧美极品| 在线精品视频一区二区| 国产精品视频成人| 国产模特精品视频久久久久| 国产精品毛片高清在线完整版| 免费观看亚洲视频大全| 欧美寡妇偷汉性猛交| 欧美成人精品在线| 欧美日韩国产精品专区 | 亚洲一区二区三区国产| 亚洲二区免费| 国产日韩综合| 国产精品高清网站| 欧美视频手机在线| 国产精品你懂得| 亚洲欧洲精品天堂一级| 亚洲国产第一| 一本色道88久久加勒比精品| 日韩一级黄色片| 午夜电影亚洲| 欧美成人精品激情在线观看| 你懂的一区二区| 国产精品高清免费在线观看| 国内精品久久久| 亚洲精品美女| 亚洲综合精品四区| 欧美高清视频免费观看| 国产精品美女久久久久久久| 激情视频一区二区| 亚洲你懂的在线视频| 欧美 日韩 国产 一区| 欧美手机在线视频| 亚洲欧洲一区| 欧美激情精品久久久久久大尺度| 国户精品久久久久久久久久久不卡| 亚洲一品av免费观看| 欧美日韩午夜激情| 在线视频日韩| 国产精品亚洲人在线观看| 亚洲女女女同性video| 国产欧美日韩在线视频| 午夜视频一区| 精品动漫3d一区二区三区| 久久精品一区二区三区中文字幕 | 狼人社综合社区| 亚洲精品一区二区三区av| 欧美日韩精品免费| 先锋影音网一区二区| 在线电影国产精品| 欧美日韩一区二区三区四区在线观看 | 午夜视频久久久久久| 国产自产2019最新不卡| 久久久久久有精品国产| 亚洲精品孕妇| 激情久久五月| 国产精品成人免费精品自在线观看| 欧美一级成年大片在线观看| 一区二区三区在线高清| 欧美视频久久| 欧美激情在线狂野欧美精品| 欧美一区二区三区在线| 亚洲精品社区| 一区免费视频| 国产精品午夜春色av| 欧美成人精品1314www| 久久精品国产精品亚洲综合 | 中文欧美字幕免费| 在线电影国产精品| 狠狠综合久久av一区二区小说|