chinese直男口爆体育生外卖, 99久久er热在这里只有精品99, 又色又爽又黄18禁美女裸身无遮挡, gogogo高清免费观看日本电视,私密按摩师高清版在线,人妻视频毛茸茸,91论坛 兴趣闲谈,欧美 亚洲 精品 8区,国产精品久久久久精品免费

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

為什么C++單例模式不能直接全部使用static變量和static函數(shù)呢?

Linux愛好者 ? 來源:Linux愛好者 ? 作者:Linux愛好者 ? 2022-06-05 14:14 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

開場

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

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

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

//大概這個樣子
classSingleton{
public:
staticvoidon(){Singleton::isOn=true;}
staticvoidoff(){Singleton::isOn=false;}
staticboolstate(){returnSingleton::isOn;}
private:
staticboolisOn;
};

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

正文

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

classSingleton{
public:
staticSingleton&getInstance(){
staticSingletoninst;
returninst;
}
Singleton(constSingleton&)=delete;
Singleton&operator=(constSingleton&)=delete;

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

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

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

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

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

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

classSingleton{
public:
staticSingleton&getInstance(){
returninst;
}
Singleton(constSingleton&)=delete;
Singleton&operator=(constSingleton&)=delete;

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

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

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


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

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

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

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

這時候兩個單例在不同文件中各自實現(xiàn),很有可能在日志管理器的單例使用配置文件單例的時候,配置文件的單例對象是沒有被初始化的。這個未初始化可能產(chǎn)生的風險指的是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ù)開始運行之后肯定是都能做完初始化的。所以如果你是在main()函數(shù)運行之后,用日志管理器的單例訪問配置文件的單例,那么其實也是沒有問題的… 這就引出Meyers' Singleton的第二個優(yōu)勢,那就是當產(chǎn)生繼承的時候。如果出現(xiàn)繼承,這種寫法中:

classSingleton{
public:
staticvoidon(){Singleton::isOn=true;}
staticvoidoff(){Singleton::isOn=false;}
staticboolstate(){returnSingleton::isOn;}
private:
staticboolisOn;
};

classMonitor:publicSingleton{
public:
staticvoidaddBrightness(intval){brightness+=val;}
staticvoidsubBrightness(intval){brightness-=val;}
staticintgetBrightness(){returnbrightness;}

private:
staticintbrightness;
};

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

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

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

審核編輯 :李倩


聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 函數(shù)
    +關(guān)注

    關(guān)注

    3

    文章

    4405

    瀏覽量

    66792
  • C++
    C++
    +關(guān)注

    關(guān)注

    22

    文章

    2121

    瀏覽量

    76684

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

文章出處:【微信號:LinuxHub,微信公眾號:Linux愛好者】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評論

    相關(guān)推薦
    熱點推薦

    常用變量的介紹

    extern:用在全局變量上表示該變量在其他文件中已經(jīng)定義;用在函數(shù)上作用同全局變量; static:用在全局
    發(fā)表于 11-21 07:05

    【CPKCOR-RA8D1】關(guān)于AI人臉檢測移植遇到的一些問題

    : 兩個不同文件實現(xiàn)了同名全局函數(shù) handle_error(但參數(shù)/簽名可能不同)。鏈接器不能接受重復(fù)符號。 修復(fù)策略: 統(tǒng)一保留單一全局實現(xiàn)(放在 common_util.c),并在其他文件中引用
    發(fā)表于 10-31 13:39

    ESD和EOS失效模式介紹

    ESD(Electro Static Discharge靜電釋放)與EOS(Electrical Over Stress 過度電性應(yīng)力)都是與電壓過應(yīng)力有關(guān)的概念,但它們之間有明顯的差異。
    的頭像 發(fā)表于 10-23 14:13 ?681次閱讀
    ESD和EOS失效<b class='flag-5'>模式</b>介紹

    PLC中Static和Temp變量的區(qū)別

    大家好,收到粉絲投稿,讓博主講下Static變量和Temp變量的區(qū)別,新入行的兄弟可能會對這兩個概念不太能理解。
    的頭像 發(fā)表于 09-24 14:51 ?873次閱讀
    PLC中<b class='flag-5'>Static</b>和Temp<b class='flag-5'>變量</b>的區(qū)別

    itop-3568開發(fā)板驅(qū)動開發(fā)指南-實驗程序的編寫

    );//傳遞字符串類型的變量 str1 12、static int __init parameter_init(void)//驅(qū)動入口函數(shù) 13、{ 14、static int i;
    發(fā)表于 05-19 10:26

    嵌入式學習-飛凌嵌入式ElfBoard ELF 1板卡-I2C設(shè)備驅(qū)動之I2C驅(qū)動構(gòu)建流程

    函數(shù),處理設(shè)備初始化和配置 // ... return 0;} static int my_i2c_remove(struct i2c_client *client){ // I2
    發(fā)表于 04-15 10:37

    飛凌嵌入式ElfBoard ELF 1板卡-I2C設(shè)備驅(qū)動之I2C驅(qū)動構(gòu)建流程

    函數(shù),處理設(shè)備初始化和配置 // ... return 0;} static int my_i2c_remove(struct i2c_client *client){ // I2
    發(fā)表于 04-15 10:21

    static在單片機中的妙用

    不要從字面意思誤以為關(guān)鍵字 static 很安靜,其實它一點都不安靜。這個關(guān)鍵字在 C 語言里主要有兩個作用,下面我們就來介紹一下 C 語言關(guān)鍵字 static 的作用,你是否 理解全
    發(fā)表于 04-02 13:50 ?1次下載

    C++學到什么程度可以找工作?

    /IP協(xié)議棧,知道如何編寫客戶端/服務(wù)器端程序。 5. **設(shè)計模式**:了解常見的設(shè)計模式,如模式、工廠
    發(fā)表于 03-13 10:19

    在starvision2上移植FreeRTOS,objdump后發(fā)現(xiàn)static變量的地址是0,怎么解決?

    變量并賦值,并對外提供kernelend符號,即在C/C++中可以調(diào)用這個量*/. = ALIGN(0x1000);PROVIDE(freememstart = .);}
    發(fā)表于 03-10 07:21

    AFE5801的static_PGA模式不能使用怎么解決?

    (讀出地址為0x04的寄存器的值),這時候能夠讀出SDOUT的值為0x030c。 當我使用static_PGA 模式讀寫的時候,設(shè)置的寄存器分別是0x000004(切換到TVG設(shè)置的寄存器組
    發(fā)表于 02-11 06:55

    AFE5801 TGC配置中采用Static PGA模式時細調(diào)增益配置的疑問求解

    對AFE5801 TGC配置時,采用Static PGA模式,對coarse_gain和fine_gain分別配置時,發(fā)現(xiàn)粗調(diào)寫進去了,但是細調(diào)增益麥斯沒寫進去。 比如:細調(diào)增益寫的是16
    發(fā)表于 02-06 08:36

    C語言如何處理函數(shù)的返回值

    的那樣,直接把 1234 賦值給了變量 ret? 搞懂這個問題不難,只要看下匯編代碼就行。 把代碼編譯一下,只編譯不鏈接,得到的就是C對應(yīng)的匯編代碼。 這塊是 test 函數(shù),不用管上
    的頭像 發(fā)表于 01-16 09:21 ?732次閱讀

    Spire.XLS for C++組件說明

    Spire.XLS for C++ 是一款專業(yè)的 C++ Excel 組件,可以用在各種 C++ 框架和應(yīng)用程序中。Spire.XLS for C++ 提供了一個對象模型 Excel
    的頭像 發(fā)表于 01-14 09:40 ?1288次閱讀
    Spire.XLS for <b class='flag-5'>C++</b>組件說明

    使用ads1294發(fā)送RDATA指令,收到的static卻是40 00 00,是什么原因?

    我使用ads1294發(fā)送RDATA指令,之前都能正確接收status(24bit)+4 channel(4*24bit)數(shù)據(jù),staticC0 00 00,但是現(xiàn)在收到的static卻是40 00 00,后面跟的數(shù)據(jù)也亂了,
    發(fā)表于 01-06 08:04