色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

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

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

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

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

Linux愛好者 ? 來源:Linux愛好者 ? 作者:Linux愛好者 ? 2022-06-05 14:14 ? 次閱讀

開場

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

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

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

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

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

正文

通過getInstance()函數(shù)獲取單例對象,這種模式的關(guān)鍵之處不是在于強(qiáng)迫你用函數(shù)來獲取對象。關(guān)鍵之處是讓static對象定義在函數(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。沒錯(cuò),也就是說這是Scott Meyers最早提出來的C++單例模式的推薦寫法。

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

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

如果是把 static對象定義成 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()再訪問,但這種不是Meyers' Singleton!


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

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

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

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

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

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

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

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

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

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

審核編輯 :李倩


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

    關(guān)注

    3

    文章

    4345

    瀏覽量

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

    關(guān)注

    22

    文章

    2114

    瀏覽量

    73792

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

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

收藏 人收藏

    評論

    相關(guān)推薦

    Spire.XLS for C++組件說明

    Spire.XLS for C++ 是一款專業(yè)的 C++ Excel 組件,可以用在各種 C++ 框架和應(yīng)用程序中。Spire.XLS for C++ 提供了一個(gè)對象模型 Excel
    的頭像 發(fā)表于 01-14 09:40 ?102次閱讀
    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

    同樣是函數(shù),在CC++中有什么區(qū)別

    ,即使沒有數(shù)據(jù)返回,也得寫 void。 第二個(gè)函數(shù)名。 C語言的函數(shù)名絕對不能重名,除了用上 weak 這樣的黑科技。同一個(gè)項(xiàng)目中,函數(shù)重名
    的頭像 發(fā)表于 11-29 10:25 ?404次閱讀

    C語言中申請的堆內(nèi)存能不能自動(dòng)釋放

    C語言中申請的堆內(nèi)存能不能自動(dòng)釋放?每次都要手動(dòng) free 太麻煩,也容易忘記。 學(xué)過 C++ 的同學(xué),應(yīng)該首先能想到智能指針。 但是這是C語言,沒有類和對象、構(gòu)造析構(gòu)這些技術(shù),想要自
    的頭像 發(fā)表于 11-27 09:33 ?160次閱讀

    C++新手容易犯的十個(gè)編程錯(cuò)誤

    簡單的總結(jié)一下?C++ 新手容易犯的一些編程錯(cuò)誤,給新人們提供一個(gè)參考。 1 有些關(guān)鍵字在 cpp 文件中多寫了 對于 C++ 類,一些關(guān)鍵字只要寫在 .h 中就好,cpp 中就不用再加上了,比如
    的頭像 發(fā)表于 11-15 12:42 ?455次閱讀

    C++中實(shí)現(xiàn)類似instanceof的方法

    函數(shù),可實(shí)際上C++中沒有。但是別著急,其實(shí)C++中有兩種簡單的方法可以實(shí)現(xiàn)類似Java中的instanceof的功能。 在 C++ 中,確定對象的類型是編程中實(shí)際需求,使開發(fā)人員
    的頭像 發(fā)表于 07-18 10:16 ?651次閱讀
    <b class='flag-5'>C++</b>中實(shí)現(xiàn)類似instanceof的方法

    wifi模式設(shè)置靜態(tài)IP地址失敗的原因?

    wifi模式在進(jìn)入連接注冊事件回調(diào)函數(shù)里調(diào)用設(shè)置靜態(tài)IP函數(shù)函數(shù)如下: static void wifi_connected_hand
    發(fā)表于 06-12 08:08

    FX2 CY7C68013A如何在C++環(huán)境中使用LoadEEPROM函數(shù)?

    我使用的是 FX2 CY7C68013A 芯片。 我知道 CyUSB.NET 庫中有我需要的 LoadEEPROM 函數(shù)。 請問如何在 C++ 環(huán)境而不是 C#/CLR 環(huán)境中使用該
    發(fā)表于 05-31 06:59

    請問頭文件能不能定義變量?

    最近在編譯一個(gè)工程的時(shí)候,突然遇到了變量重復(fù)定義的問題,根據(jù)提示打開這幾個(gè) C 文件,并沒有發(fā)現(xiàn)定義變量的地方。后來再找一找,原來變量定義在了頭文件里面。
    的頭像 發(fā)表于 04-28 09:33 ?1239次閱讀

    C語言中的三種形式變量

    局部變量是在一個(gè)函數(shù)、代碼塊內(nèi)部聲明的變量,只能被該函數(shù)或者代碼塊內(nèi)部應(yīng)用。局部變量函數(shù)之外不
    發(fā)表于 03-11 17:34 ?791次閱讀
    <b class='flag-5'>C</b>語言中的三種形式<b class='flag-5'>變量</b>

    淺談C語言中的函數(shù)定義

    如果函數(shù)要使用參數(shù),則必須聲明接受參數(shù)值的變量。這些變量稱為函數(shù)的形式參數(shù)。 形式參數(shù)就像函數(shù)內(nèi)的其他局部
    發(fā)表于 03-11 10:09 ?431次閱讀

    嵌入式C語言面試大挑戰(zhàn)

    C++static關(guān)鍵字除了具有C中的作用還有在類中的使用在類中,static可以用來修飾靜態(tài)數(shù)據(jù)成員和靜態(tài)成員方法靜態(tài)數(shù)據(jù)成員
    發(fā)表于 03-05 14:18 ?380次閱讀
    嵌入式<b class='flag-5'>C</b>語言面試大挑戰(zhàn)

    簡單總結(jié)一下嵌入式C++中常見的錯(cuò)誤形式

    對于 C++ 類,一些關(guān)鍵字只要寫在 .h 中就好,cpp 中就不用再加上了,比如 virtual、static 等關(guān)鍵字,如果在 cpp 中多寫,編譯器會報(bào)錯(cuò)。
    的頭像 發(fā)表于 02-23 09:40 ?557次閱讀

    c語言,c++,java,python區(qū)別

    C語言、C++、Java和Python是四種常見的編程語言,各有優(yōu)點(diǎn)和特點(diǎn)。 C語言: C語言是一種面向過程的編程語言。它具有底層的特性,能夠?qū)τ?jì)算機(jī)硬件進(jìn)行
    的頭像 發(fā)表于 02-05 14:11 ?2551次閱讀

    要怎么增加FLEXRAY STATIC MESSAGE ?

    我使用 SDL 7.8.0 版本,芯片是 tviibh8m 我看了 SDL 里面的 FLEXRAY 示例 comm_static_dynamic 我想要增加 STATIC MESSAGE
    發(fā)表于 02-01 08:33
    主站蜘蛛池模板: 99re1久久热在线播放 | 免费国产综合视频在线看 | 国产精品人妻一区免费看8C0M | 亚洲欧美中文字幕网站大全 | 久久这里只有精品1 | 亚洲免费网站观看视频 | 亚欧日韩毛片在线看免费网站 | 一二三四电影完整版免费观看 | 天天夜夜草草久久亚洲香蕉 | 和美女啪啪啪动态图 | 吻嘴胸全身好爽床大全 | 亚洲视频欧美在线专区 | 百性阁综合社区 | GAY2022空少被体育生暴菊 | 男男h啪肉np文总受 男男h开荤粗肉h文1v1 | 麻花传媒MD0044视频 | www.色小姐| 久久精品观看 | 狼人无码伊人AV啪啪 | 国产在线高清视频无码不卡 | 国产亚洲精品久久无亚洲 | 黄色a三级三级三级免费看 黄色a三级免费看 | 精品国产乱码久久久久久夜深人妻 | 脱女学小内内摸出水网站免费 | 国产成人精品视频播放 | 亚洲乱码中文字幕久久孕妇黑人 | 挺进绝色老师的紧窄小肉六 | 囚禁固定在调教椅上扩张H 秋霞最新高清无码鲁丝片 秋霞在线看片无码免费 | 好男人的视频在线观看 | 亚洲欧美一区二区久久 | 色美妞论坛 | 国产在线精品一区二区网站免费 | 精品无码久久久久久久久 | 一级毛片直接看 | 91在线一区二区 | 城中村快餐嫖老妇对白 | 好紧的小嫩嫩17p | 九九热在线观看视频 | 暖暖视频免费观看社区 | 亚洲AV综合色一区二区三区 | 中文字幕亚洲欧美日韩2o19 |