91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

C++單例模式實例化一個對象不全部使用static的原因是什么

發布時間:2022-05-12 11:30:42 來源:億速云 閱讀:181 作者:zzz 欄目:開發技術

今天小編給大家分享一下C++單例模式實例化一個對象不全部使用static的原因是什么的相關知識點,內容詳細,邏輯清晰,相信大部分人都還太了解這方面的知識,所以分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后有所收獲,下面我們一起來了解一下吧。

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

通過getInstance()函數獲取單例對象,這種模式的關鍵之處不是在于強迫你用函數來獲取對象。關鍵之處是讓static對象定義在函數內部,變成局部static變量。

看下這種實現方式的經典demo:

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

    // 其他數據函數
    // ...

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

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

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

所以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;

    // 其他數據函數
    // ...

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

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

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

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

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

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

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

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

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

如果出現繼承,這種寫法中:

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顯示器類有開關狀態,同時擴展了一個亮度的成員。但是父子類的static成員變量是共享的,其isOn成員會有問題。

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

以上就是“C++單例模式實例化一個對象不全部使用static的原因是什么”這篇文章的所有內容,感謝各位的閱讀!相信大家閱讀完這篇文章都有很大的收獲,小編每天都會為大家更新不同的知識,如果還想學習更多的知識,請關注億速云行業資訊頻道。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

滦南县| 个旧市| 西贡区| 邻水| 北海市| 永胜县| 红河县| 柘城县| 闽侯县| 鸡东县| 黎平县| 岑溪市| 乐平市| 德令哈市| 龙川县| 仙居县| 南漳县| 泗水县| 四川省| 内江市| 辽阳县| 勃利县| 商河县| 怀化市| 师宗县| 宜阳县| 信丰县| 富民县| 贺兰县| 乐平市| 通化县| 响水县| 上林县| 雷州市| 南溪县| 遵化市| 治县。| 德钦县| 大同县| 扎鲁特旗| 张掖市|