詳解C++編程中類的成員變量和成員函數(shù)的相關(guān)知識(shí)
C++類的成員變量和成員函數(shù)
類是一種數(shù)據(jù)類型,它類似于普通的數(shù)據(jù)類型,但是又有別于普通的數(shù)據(jù)類型。類這種數(shù)據(jù)類型是一個(gè)包含成員變量和成員函數(shù)的一個(gè)集合。
類的成員變量和普通變量一樣,也有數(shù)據(jù)類型和名稱,占用固定長度的內(nèi)存空間。但是,在定義類的時(shí)候不能對(duì)成員變量賦值,因?yàn)轭愔皇且环N數(shù)據(jù)類型,本身不占用內(nèi)存空間,而變量的值則需要內(nèi)存來存儲(chǔ)。
類的成員函數(shù)也和普通函數(shù)一樣,都有返回值和參數(shù)列表,它與一般函數(shù)的區(qū)別是:成員函數(shù)是一個(gè)類的成員,出現(xiàn)在類體中,它的作用范圍由類來決定;而普通函數(shù)是獨(dú)立的,作用范圍是全局的,或位于某個(gè)命名空間內(nèi)。
上節(jié)我們?cè)谧詈蟮耐暾纠薪o出了 Student 類的定義,如下所示:
class Student{ public: //類包含的變量 char *name; int age; float score; public: //類包含的函數(shù) void say(){ printf("%s的年齡是 %d,成績是 %f\n", name, age, score); } }; 上面的代碼在類體中定義了成員函數(shù)。你也可以只在類體中聲明函數(shù),而將函數(shù)定義放在類體外面,如下圖所示: class Student{ public: char *name; int age; float score; public: void say(); //函數(shù)聲明 }; //函數(shù)定義 void Student::say(){ printf("%s的年齡是 %d,成績是 %f\n", name, age, score); }
在類體中直接定義函數(shù)時(shí),不需要在函數(shù)名前面加上類名,因?yàn)楹瘮?shù)屬于哪一個(gè)類是不言而喻的。
但當(dāng)成員函數(shù)定義在類外時(shí),就必須在函數(shù)名前面加上類名予以限定。::被稱為域解析符(也稱作用域運(yùn)算符或作用域限定符),用來連接類名和函數(shù)名,指明當(dāng)前函數(shù)屬于哪個(gè)類。
如果在域解析符“::”的前面沒有類名,或者函數(shù)名前面既無類名又無域解析符“::”,如:
//無類名 ::say( ){ //TODO } //無類名也無域解析符 say( ){ //TODO }
則表示 say() 函數(shù)不屬于任何類,這個(gè)函數(shù)不是成員函數(shù),而是全局函數(shù),即非成員函數(shù)的一般普通函數(shù)。
成員函數(shù)必須先在類體中作原型聲明,然后在類外定義,也就是說類體的位置應(yīng)在函數(shù)定義之前,否則編譯時(shí)會(huì)出錯(cuò)。
雖然成員函數(shù)在類的外部定義,但在調(diào)用時(shí)會(huì)根據(jù)在類中聲明的函數(shù)原型找到函數(shù)的定義(函數(shù)代碼),從而執(zhí)行該函數(shù)。
inline 成員函數(shù)
在類體中和類體外定義成員函數(shù)是有區(qū)別的:在類體中定義的成員函數(shù)為內(nèi)聯(lián)(inline)函數(shù),在類體外定義的不是。
內(nèi)聯(lián)函數(shù)一般不是我們所期望的,它會(huì)將函數(shù)調(diào)用處用函數(shù)體替代,所以我建議在類體內(nèi)部對(duì)成員函數(shù)作聲明,而在類體外部進(jìn)行定義,這是一種良好的編程習(xí)慣。
當(dāng)然,如果你的函數(shù)比較短小,希望定義為內(nèi)聯(lián)函數(shù),那也沒有什么不妥的。
如果你既希望將函數(shù)定義在類體外部,又希望它是內(nèi)聯(lián)函數(shù),那么可以在聲明函數(shù)時(shí)加 inline 關(guān)鍵字,如下所示:
class Student{ public: char *name; int age; float score; public: inline void say(); //聲明為內(nèi)聯(lián)函數(shù) }; //函數(shù)定義 void Student::say(){ printf("%s的年齡是 %d,成績是 %f\n", name, age, score); }
這樣,say() 就會(huì)變成內(nèi)聯(lián)函數(shù)。
在類體內(nèi)部定義的函數(shù)也可以加 inline 關(guān)鍵字,但這是多余的,因?yàn)轭愺w內(nèi)部定義的函數(shù)默認(rèn)就是內(nèi)聯(lián)函數(shù)。
值得注意的是,如果在類體外定義 inline 函數(shù),則必須將類定義和成員函數(shù)的定義都放在同一個(gè)頭文件中(或者寫在同一個(gè)源文件中),否則編譯時(shí)無法進(jìn)行嵌入(將函數(shù)代碼的嵌入到函數(shù)調(diào)用出)。這樣做雖然提高了程序的執(zhí)行效率,但從軟件工程質(zhì)量的角度來看,這樣做并不是好的辦法,因此實(shí)際開發(fā)中較少在類中使用內(nèi)聯(lián)函數(shù)。
C++提出內(nèi)聯(lián)函數(shù)的主要用意是:用內(nèi)聯(lián)函數(shù)取代帶參宏定義(函數(shù)傳參比宏更加方便易用),而不是提高程序運(yùn)行效率,因?yàn)榕c執(zhí)行函數(shù)花費(fèi)的時(shí)間相比,調(diào)用函數(shù)花費(fèi)的時(shí)間往往微乎其微。
C++成員函數(shù)的存儲(chǔ)方式
用類去定義對(duì)象時(shí),系統(tǒng)會(huì)為每一個(gè)對(duì)象分配存儲(chǔ)空間。如果一個(gè)類包括了數(shù)據(jù)和函數(shù),要分別為數(shù)據(jù)和函數(shù)的代碼分配存儲(chǔ)空間。
按理說,如果用同一個(gè)類定義了10個(gè)對(duì)象,那么就需要分別為10個(gè)對(duì)象的數(shù)據(jù)和函數(shù)代碼分配存儲(chǔ)單元,如下圖所示。
能否只用一段空間來存放這個(gè)共同的函數(shù)代碼段,在調(diào)用各對(duì)象的函數(shù)時(shí),都去調(diào)用這個(gè)公用的函數(shù)代碼。如圖所示。
顯然,這樣做會(huì)大大節(jié)約存儲(chǔ)空間。C++編譯系統(tǒng)正是這樣做的,因此每個(gè)對(duì)象所占用的存儲(chǔ)空間只是該對(duì)象的數(shù)據(jù)部分所占用的存儲(chǔ)空間,而不包括函數(shù)代碼所占用的存儲(chǔ)空間。如果聲明了一個(gè)類:
class Time { public: int hour; int minute; int sec; void set( ) { cin>>a>>b>>c; } };
可以用下面的語句來輸出該類對(duì)象所占用的字節(jié)數(shù):
cout<<sizeof(Time)<<endl;
輸出的值是12。
這就證明了一個(gè)對(duì)象所占的空間大小只取決于該對(duì)象中數(shù)據(jù)成員所占的空間,而與成員函數(shù)無關(guān)。
函數(shù)代碼是存儲(chǔ)在對(duì)象空間之外的。如果對(duì)同一個(gè)類定義了10個(gè)對(duì)象,這些對(duì)象的成員函數(shù)對(duì)應(yīng)的是同一個(gè)函數(shù)代碼段,而不是10個(gè)不同的函數(shù)代碼段。需要注意的是,雖然調(diào)用不同對(duì)象的成員函數(shù)時(shí)都是執(zhí)行同一段函數(shù)代碼,但是執(zhí)行結(jié)果一般是不相同的。
不同的對(duì)象使用的是同一個(gè)函數(shù)代碼段,它怎么能夠分別對(duì)不同對(duì)象中的數(shù)據(jù)進(jìn)行操作呢?
原來C++為此專門設(shè)立了一個(gè)名為this的指針,用來指向不同的對(duì)象。需要說明:
不論成員函數(shù)在類內(nèi)定義還是在類外定義,成員函數(shù)的代碼段都用同一種方式存儲(chǔ)。
不要將成員函數(shù)的這種存儲(chǔ)方式和inMne(內(nèi)置)函數(shù)的概念混淆。不要誤以為用inline聲明(或默認(rèn)為inline)的成員函數(shù),其代碼段占用對(duì)象的存儲(chǔ)空間,而不用 inline聲明的成員函數(shù),其代碼段不占用對(duì)象的存儲(chǔ)空間。不論是否用inline聲明,成員函數(shù)的代碼段都不占用對(duì)象的存儲(chǔ)空間。用inline聲明的作用是在調(diào)用該函數(shù)時(shí),將函數(shù)的代碼段復(fù)制插人到函數(shù)調(diào)用點(diǎn),而若不用inline聲明,在調(diào)用該函數(shù)時(shí),流程轉(zhuǎn)去函數(shù)代碼段的人口地址,在執(zhí)行完該函數(shù)代碼段后,流程返回函數(shù)調(diào)用點(diǎn)。inline與成員函數(shù)是否占用對(duì)象的存儲(chǔ)空間無關(guān),它們不屬同一個(gè)問題,不應(yīng)搞混。
應(yīng)當(dāng)說明,常說的“某某對(duì)象的成員函數(shù)”,是從邏輯的角度而言的,而成員函數(shù)的存儲(chǔ)方式,是從物理的角度而言的,二者是不矛盾的。
欄 目:C語言
下一篇:深入解析C++編程中的靜態(tài)成員函數(shù)
本文標(biāo)題:詳解C++編程中類的成員變量和成員函數(shù)的相關(guān)知識(shí)
本文地址:http://mengdiqiu.com.cn/a1/Cyuyan/2713.html
您可能感興趣的文章
- 04-02c語言沒有round函數(shù) round c語言
- 01-10求子數(shù)組最大和的解決方法詳解
- 01-10深入二叉樹兩個(gè)結(jié)點(diǎn)的最低共同父結(jié)點(diǎn)的詳解
- 01-10數(shù)據(jù)結(jié)構(gòu)課程設(shè)計(jì)- 解析最少換車次數(shù)的問題詳解
- 01-10數(shù)據(jù)結(jié)構(gòu)課程設(shè)計(jì)-用棧實(shí)現(xiàn)表達(dá)式求值的方法詳解
- 01-10HDOJ 1443 約瑟夫環(huán)的最新應(yīng)用分析詳解
- 01-10深入理解C++中常見的關(guān)鍵字含義
- 01-10使用C++實(shí)現(xiàn)全排列算法的方法詳解
- 01-10如何查看進(jìn)程實(shí)際的內(nèi)存占用情況詳解
- 01-10深入Main函數(shù)中的參數(shù)argc,argv的使用詳解


閱讀排行
本欄相關(guān)
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言的正則匹配函數(shù) c語言正則表達(dá)
- 04-02c語言用函數(shù)寫分段 用c語言表示分段
- 04-02c語言中對(duì)數(shù)函數(shù)的表達(dá)式 c語言中對(duì)
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段
- 04-02C語言中怎么打出三角函數(shù) c語言中怎
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求
隨機(jī)閱讀
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 04-02jquery與jsp,用jquery
- 01-10delphi制作wav文件的方法
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 01-10使用C語言求解撲克牌的順子及n個(gè)骰子
- 01-10C#中split用法實(shí)例總結(jié)
- 01-11ajax實(shí)現(xiàn)頁面的局部加載
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什