国产精品久久久aaaa,日日干夜夜操天天插,亚洲乱熟女香蕉一区二区三区少妇,99精品国产高清一区二区三区,国产成人精品一区二区色戒,久久久国产精品成人免费,亚洲精品毛片久久久久,99久久婷婷国产综合精品电影,国产一区二区三区任你鲁

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

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

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

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

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

掃碼添加小助手

加入工程師交流群

開(kāi)場(chǎng)

前段時(shí)間在知乎回答了這樣一個(gè)問(wèn)題:

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

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

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

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

正文

通過(guò)getInstance()函數(shù)獲取單例對(duì)象,這種模式的關(guān)鍵之處不是在于強(qiáng)迫你用函數(shù)來(lái)獲取對(duì)象。關(guān)鍵之處是讓static對(duì)象定義在函數(shù)內(nèi)部,變成局部static變量。看下這種實(shí)現(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ù)成員
//...
};

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

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

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

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

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

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

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

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


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

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

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

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

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

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

講到這,我們對(duì)Meyers' Singleton的盲目鼓吹也需冷靜一下,因?yàn)镃++同樣能保證所有文件內(nèi)(非函數(shù)內(nèi))的static變量在main()函數(shù)開(kāi)始運(yùn)行之后肯定是都能做完初始化的。所以如果你是在main()函數(shù)運(yùn)行之后,用日志管理器的單例訪問(wèn)配置文件的單例,那么其實(shí)也是沒(méi)有問(wèn)題的… 這就引出Meyers' Singleton的第二個(gè)優(yōu)勢(shì),那就是當(dāng)產(chǎn)生繼承的時(shí)候。如果出現(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;
};

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

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

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

審核編輯 :李倩


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

    關(guān)注

    3

    文章

    4417

    瀏覽量

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

    關(guān)注

    22

    文章

    2123

    瀏覽量

    77110

原文標(biāo)題:C++ 的單例模式為什么不直接全部使用 static,而是非要實(shí)例化一個(gè)對(duì)象?

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

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

掃碼添加小助手

加入工程師交流群

    評(píng)論

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

    keil實(shí)現(xiàn)cc++混合編程

    參考touchgfx生成的代碼,發(fā)現(xiàn)了一個(gè)不需要添加--cpp11 參數(shù)的解決方法,具體操作如下。 一、創(chuàng)建一個(gè)空白的C文件和頭文件在頭文件中定義c++文件中需要調(diào)用的函數(shù),如圖所示 二、在
    發(fā)表于 01-26 08:58

    keil定義fputc函數(shù)

    中統(tǒng)一調(diào)用。 // 設(shè)備文件寫入一個(gè)字符 static int fputc_dev(int c,FILE * stream) { const libc_device_file *dev
    發(fā)表于 01-22 08:25

    嵌入式C語(yǔ)言中各變量存儲(chǔ)位置

    局部變量、局部靜態(tài)變量、全局變量、全局靜態(tài)變量區(qū)別如下: 局部變量: 棧區(qū); 局部靜態(tài)變量:靜
    發(fā)表于 12-25 07:54

    C語(yǔ)言與C++的區(qū)別及聯(lián)系

    創(chuàng)建源文件時(shí)什么都不給,默認(rèn)是.cpp。 3、返回值 C語(yǔ)言中,如果一個(gè)函數(shù)沒(méi)有指定返回值類型,默認(rèn)返回int類型;C++中,如果一個(gè)函數(shù)沒(méi)有返回值則必須指定為void。 4、參
    發(fā)表于 12-24 07:23

    如何搞定嵌入式 C語(yǔ)言中的全局變量問(wèn)題?

    時(shí)序,起承轉(zhuǎn)合。但是盡量不要用來(lái)傳遞參數(shù),這個(gè)很忌諱的。 2、盡量把變量的作用范圍控制在使用它的模塊里面,如果其他模塊要訪問(wèn),就開(kāi)個(gè)讀或?qū)?b class='flag-5'>函數(shù)接口出來(lái),嚴(yán)格控制訪問(wèn)范圍。這一點(diǎn),C++的private
    發(fā)表于 12-16 06:54

    C語(yǔ)言全局變量重點(diǎn)使用

    全局變量絕不會(huì)位于寄存器中。使用指針或者函數(shù)調(diào)用,可以直接修改全局變量的值。 因此,編譯器不能將全局變量
    發(fā)表于 12-12 06:58

    CC++之間的聯(lián)系

    1、語(yǔ)法兼容性: C++完全兼容C語(yǔ)言的語(yǔ)法,這意味著任何有效的C語(yǔ)言程序都可以直接C++編譯器下編譯通過(guò)。 2、底層控制:
    發(fā)表于 12-11 06:51

    C語(yǔ)言和C++之間的區(qū)別是什么

    區(qū)別 1、面向?qū)ο缶幊?(OOP): C語(yǔ)言是一種面向過(guò)程的語(yǔ)言,它強(qiáng)調(diào)的是通過(guò)函數(shù)將任務(wù)分解為一系列步驟進(jìn)行執(zhí)行。 C++C語(yǔ)言的基礎(chǔ)上擴(kuò)展了面向?qū)ο蟮奶匦裕С诸?class)
    發(fā)表于 12-11 06:23

    常用變量的介紹

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

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

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

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

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

    itop-3568開(kāi)發(fā)板驅(qū)動(dòng)開(kāi)發(fā)指南-實(shí)驗(yàn)程序的編寫

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

    static在單片機(jī)中的妙用

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

    C++學(xué)到什么程度可以找工作?

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

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

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