我們之前在 C 語(yǔ)言中可以定義 const 成員,那么我們是否可以在類(lèi)中定義 const 成員呢?我們來(lái)看看下面代碼中的類(lèi)定義是否合法呢?如果合法,ci 的值是什么,存儲(chǔ)在哪里呢?
創(chuàng)新互聯(lián)公司主要從事網(wǎng)站建設(shè)、網(wǎng)站設(shè)計(jì)、網(wǎng)頁(yè)設(shè)計(jì)、企業(yè)做網(wǎng)站、公司建網(wǎng)站等業(yè)務(wù)。立足成都服務(wù)太原,10年網(wǎng)站建設(shè)經(jīng)驗(yàn),價(jià)格優(yōu)惠、服務(wù)專(zhuān)業(yè),歡迎來(lái)電咨詢(xún)建站服務(wù):028-86922220#includeclass Test { private: const int ci; public: int getCI() { return ci; } }; int main() { Test t; printf("t.ci = %d\n", t.getCI()); return 0; }
照我們之前學(xué)習(xí)的知識(shí)可以猜測(cè)對(duì)象 t 是創(chuàng)建于棧上,那么它中的 ci 便會(huì)是隨機(jī)值了。我們看看編譯結(jié)果
編譯報(bào)錯(cuò),它說(shuō)我們沒(méi)有初始化 ci,下面我們?cè)谒x的時(shí)候并初始化為10。也就是將第 6 行改為 const int ci = 10;我們?cè)俅慰纯此欠窬幾g通過(guò)
我們看到編譯產(chǎn)生警告了,雖然它的運(yùn)行結(jié)果是正確的。我們之前說(shuō)過(guò),一個(gè)優(yōu)秀的程序員會(huì)將任何一個(gè)警告都看作是錯(cuò)誤,因?yàn)樗慕Y(jié)果將是不確定的。這樣寫(xiě)是由問(wèn)題的,洽好 g++ 編譯器支持這樣寫(xiě),我們不能寫(xiě)出依賴(lài)于某種編譯器特性的代碼。所以這時(shí)便會(huì)用到初始化列表了,在 C++ 中提供了初始化列表對(duì)成員變量進(jìn)行初始化,它的語(yǔ)法規(guī)則如下
那么我們?cè)诔绦蛑屑由蠘?gòu)造函數(shù)并用初始化列表進(jìn)行初始化,如下
Test() : ci(5) { }
我們?cè)俅尉幾g,如下
我們看到編譯器沒(méi)有報(bào)任何警告,這便證明了在 C++ 中提供了初始化列表對(duì)成員變量進(jìn)行初始化。那么我們?cè)谶@塊有幾個(gè)注意事項(xiàng):a> 成員的初始化順序與成員的聲明順序相同;b> 成員的初始化順序與初始化列表中的位置無(wú)關(guān);c> 初始化列表先于構(gòu)造函數(shù)的函數(shù)體執(zhí)行。為例更加形象的說(shuō)明,我們?cè)俅我源a為例進(jìn)行說(shuō)明
#includeclass Value { private: int mi; public: Value(int i) { printf("i = %d\n", i); mi = i; } int getMI() { return mi; } }; class Test { private: Value m2; Value m3; Value m1; public: Test() : m1(1), m2(2), m3(3) { printf("Test::Test()\n"); } }; int main() { Test t; return 0; }
我們?cè)?Test 類(lèi)中定義了 3 個(gè) Value 對(duì)象,然后在構(gòu)造函數(shù)中使用了初始化列表對(duì)他們進(jìn)行初始化。在構(gòu)造函數(shù)中我們加入了一條打印語(yǔ)句,按照我們之前講的,應(yīng)該在初始化完了之后再打印那條語(yǔ)句。我們編譯看看結(jié)果
我們看到打印的是如我們分析的那樣,但是前面初始化的順序好像不太一樣,我們是按照 m1、m2、m3 這樣的順序進(jìn)行初始化的。想想我們之前說(shuō)的:初始化的順序和它聲明的順序相同,和它的位置并無(wú)關(guān)。所以看看我們聲明順序就知道打印的是正確的了。
類(lèi)中的 const 成員會(huì)被分配空間的,它的本質(zhì)是只讀變量并且只能在初始化列表中指定初始值。編譯器無(wú)法直接得到 const 成員的初始值,因此無(wú)法進(jìn)入符號(hào)表成為真正意義上的常量。我們以代碼為例進(jìn)行說(shuō)明
#includeclass Test { private: const int ci; public: Test() : ci(10) { printf("Test::Test()\n"); } int getCI() { return ci; } int setCI(int v) { int* p = const_cast (&ci); *p = v; } }; int main() { Test t; printf("t.ci = %d\n", t.getCI()); t.setCI(100); printf("t.ci = %d\n", t.getCI()); return 0; }
我們看看編譯結(jié)果是否改變了 ci 的值
我們看到已經(jīng)成功的通過(guò)指針 + const_cast去掉了它的 const 屬性,也就是說(shuō),它只是一個(gè)具有只讀屬性的變量。我們?cè)俅螐?qiáng)調(diào)下,初始化與賦值不同。初始化是對(duì)正在創(chuàng)建的對(duì)象進(jìn)行初值設(shè)置,而賦值則是對(duì)已經(jīng)存在的對(duì)象進(jìn)行值設(shè)置。通過(guò)對(duì)初始化列表的學(xué)習(xí),總結(jié)如下:1、類(lèi)中可以使用初始化列表對(duì)成員進(jìn)行初始化;2、初始化列表先于構(gòu)造函數(shù)體執(zhí)行;3、類(lèi)中可以定義 const 成員變量,const 成員變量必須在初始化列表中指定初值,const 成員變量為只讀變量。
歡迎大家一起來(lái)學(xué)習(xí) C++ 語(yǔ)言,可以加我QQ:243343083。
另外有需要云服務(wù)器可以了解下創(chuàng)新互聯(lián)scvps.cn,海內(nèi)外云服務(wù)器15元起步,三天無(wú)理由+7*72小時(shí)售后在線(xiàn),公司持有idc許可證,提供“云服務(wù)器、裸金屬服務(wù)器、高防服務(wù)器、香港服務(wù)器、美國(guó)服務(wù)器、虛擬主機(jī)、免備案服務(wù)器”等云主機(jī)租用服務(wù)以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡(jiǎn)單易用、服務(wù)可用性高、性?xún)r(jià)比高”等特點(diǎn)與優(yōu)勢(shì),專(zhuān)為企業(yè)上云打造定制,能夠滿(mǎn)足用戶(hù)豐富、多元化的應(yīng)用場(chǎng)景需求。