C++中Struct和Class的区别
Struct和Class的區(qū)別
今天這篇博文主要講解在C++中關(guān)鍵字struct和class的區(qū)別。這篇博文,將會系統(tǒng)的將這兩個關(guān)鍵字的不同面進(jìn)行詳細(xì)的講解。
從語法上來講,class和struct做類型定義時只有兩點(diǎn)區(qū)別:
但是在C++中,struct進(jìn)行了擴(kuò)展,現(xiàn)在它已經(jīng)不僅僅是一個包含不同數(shù)據(jù)類型的數(shù)據(jù)結(jié)構(gòu)了,它包括了更多的功能。
Struct能包含成員函數(shù)嗎?
是的,答案是肯定的。現(xiàn)在就讓我寫一段代碼驗(yàn)證一下:
/* ** FileName : StructAndClassDiffDemo ** Author : Jelly Young ** Date : 2013/12/7 ** Description : More information, please go to http://www.jellythink.com */#include <iostream> using namespace std;struct Test {int a;int getA(){return a;}void setA(int temp){a = temp;} };int main(int argc, char* argv[]) {Test testStruct;testStruct.setA(10);cout<<"Get the value from struct:"<<testStruct.getA()<<endl;Test *testStructPointer = new Test;testStructPointer->setA(20);cout<<"Get the value from struct again:"<<testStructPointer->getA()<<endl;delete testStructPointer;return 0; }以上的代碼會很正確的運(yùn)行,是的;沒錯,struct能包含成員函數(shù)的。
Struct有自己的構(gòu)造函數(shù)嗎?
是的,可以的。看以下測試代碼:
/* ** FileName : StructAndClassDiffDemo ** Author : Jelly Young ** Date : 2013/12/7 ** Description : More information, please go to http://www.jellythink.com */#include <iostream> using namespace std;struct Test {int a;Test(){a = 100;}int getA(){return a;}void setA(int temp){a = temp;} };int main(int argc, char* argv[]) {Test testStruct;testStruct.setA(10);cout<<"Get the value from struct:"<<testStruct.getA()<<endl; Test *testStructPointer = new Test; testStructPointer->setA(20);cout<<"Get the value from struct again:"<<testStructPointer->getA()<<endl;delete testStructPointer;// test the constructorTest testConstructor;cout<<"Set the value by the construct and get it:"<<testConstructor.getA()<<endl;return 0; }Struct可以有析構(gòu)函數(shù)么?
讓我來驗(yàn)證一下:
/* ** FileName : StructAndClassDiffDemo ** Author : Jelly Young ** Date : 2013/12/7 ** Description : More information, please go to http://www.jellythink.com */#include <iostream> using namespace std;struct Test {int a;Test(){a = 100;}int getA(){return a;}void setA(int temp){a = temp;}~Test(){cout<<"Destructor function called."<<endl;} };int main(int argc, char* argv[]) {Test testStruct;testStruct.setA(10);cout<<"Get the value from struct:"<<testStruct.getA()<<endl; Test *testStructPointer = new Test; testStructPointer->setA(20);cout<<"Get the value from struct again:"<<testStruct.getA()<<endl;delete testStructPointer;// test the constructorTest testConstructor;cout<<"Set the value by the construct and get it:"<<testConstructor.getA()<<endl;return 0; }是的,完全支持析構(gòu)函數(shù)。
Struct支持繼承么?
再讓我寫代碼驗(yàn)證一下:
/* ** FileName : StructAndClassDiffDemo ** Author : Jelly Young ** Date : 2013/12/7 ** Description : More information, please go to http://www.jellythink.com */#include <iostream> using namespace std;struct A {int a;A(){a = 10;}void print(){cout<<"I am from A"<<endl;} };struct B : A {int b;B(){a = 30; // set a to 30b = 20;}/*void print(){cout<<"I am from B"<<endl;}*/ };int main(int argc, char* argv[]) {B b1;cout<<b1.a<<endl;cout<<b1.b<<endl;b1.print();A a1;cout<<a1.a<<endl;a1.print();return 0; }運(yùn)行上述代碼,struct支持繼承。
Struct支持多肽么?
寫代碼測試一下便知:
/* ** FileName : StructAndClassDiffDemo ** Author : Jelly Young ** Date : 2013/12/7 ** Description : More information, please go to http://www.jellythink.com */#include <iostream> using namespace std;struct A {virtual void print() = 0; };struct B : A {void print(){cout<<"I am from B"<<endl;} };struct C : A {void print(){cout<<"I am from C"<<endl; } }; int main(int argc, char* argv[]) { A *a1; B *b1 = new B; C *c1 = new C; a1 = b1; a1->print(); // call B, not Aa1 = c1;a1->print(); // call C, not Areturn 0; }Struct支持Private、Protected和Public關(guān)鍵字么?
/* ** FileName : StructAndClassDiffDemo ** Author : Jelly Young ** Date : 2013/12/7 ** Description : More information, please go to http://www.jellythink.com */#include <iostream> using namespace std;struct A { private:int b;protected:int c;public:A(){b = 10;c = 20;d = 30;}int d; };struct B : A {void printA_C(){cout<<A::c<<endl;};// private member can not see/*void printA_B(){cout<<A::b<<endl;}*/void printA_D(){cout<<A::d<<endl;} };int main(int argc, char* argv[]) {A a1;B b1;// private member can not see//cout<<a1.b<<endl;// protected member can not see//cout<<a1.c<<endl;// public member can seecout<<a1.d<<endl;return 0; }寫了這么多了,那么會出現(xiàn)這種一個狀況,如果是class的父類是struct關(guān)鍵字描述的,那么默認(rèn)訪問屬性是什么?
當(dāng)出現(xiàn)這種情況時,到底默認(rèn)是public繼承還是private繼承,取決于子類而不是基類。class可以繼承自struct修飾的類;同時,struct也可以繼承自class修飾的類,繼承屬性如下列描述:
struct A{}; class B:A{}; // private 繼承class A{}; struct B:A{}; // public 繼承最后,那么到底是使用struct,還是使用class呢?這個看個人喜好,但是這里有一個編程規(guī)范的問題,當(dāng)你覺得你要做的更像是一種數(shù)據(jù)結(jié)構(gòu)的話,那么用struct,如果你要做的更像是一種對象的話,那么用class。
《無人區(qū)》正在熱映。
2013年12月7日 于大連,東軟。
====================================================================================
C++中的struct對C中的struct進(jìn)行了擴(kuò)充,它已經(jīng)不再只是一個包含不同數(shù)據(jù)類型的數(shù)據(jù)結(jié)構(gòu)了,它已經(jīng)獲取了太多的功能。
struct能包含成員函數(shù)嗎??能!
struct能繼承嗎??能!!
struct能實(shí)現(xiàn)多態(tài)嗎??能!!!?
既然這些它都能實(shí)現(xiàn),那它和class還能有什么區(qū)別?
最本質(zhì)的一個區(qū)別就是默認(rèn)的訪問控制:?
默認(rèn)的繼承訪問權(quán)限
struct是public的,class是private的。
你可以寫如下的代碼:
struct?A
{
??char?a;
};
struct?B?:?A
{
??char?b;
};
這個時候B是public繼承A的。
如果都將上面的struct改成class,那么B是private繼承A的。這就是默認(rèn)的繼承訪問權(quán)限。?
所以我們在平時寫類繼承的時候,通常會這樣寫:
class?B?:?public?A
就是為了指明是public繼承,而不是用默認(rèn)的private繼承。
?
當(dāng)然,到底默認(rèn)是public繼承還是private繼承,取決于子類而不是基類。
我的意思是,struct可以繼承class,同樣class也可以繼承struct,那么默認(rèn)的繼承訪問權(quán)限是看子類到底是用的struct還是class。如下:
?
struct?A{};class?B?:?A{};?//private繼承
struct?C?:?B{};?//public繼承
?
struct作為數(shù)據(jù)結(jié)構(gòu)的實(shí)現(xiàn)體,它默認(rèn)的數(shù)據(jù)訪問控制是public的,而class作為對象的實(shí)現(xiàn)體,它默認(rèn)的成員變量訪問控制是private的
?
我依舊強(qiáng)調(diào)struct是一種數(shù)據(jù)結(jié)構(gòu)的實(shí)現(xiàn)體,雖然它是可以像class一樣的用。我依舊將struct里的變量叫數(shù)據(jù),class內(nèi)的變量叫成員,雖然它們并無區(qū)別。
到底是用struct還是class,完全看個人的喜好,你可以將程序里所有的class全部替換成struct,它依舊可以很正常的運(yùn)行。但我給出的最?好建議,還是:當(dāng)你覺得你要做的更像是一種數(shù)據(jù)結(jié)構(gòu)的話,那么用struct,如果你要做的更像是一種對象的話,那么用class。?
當(dāng)然,我在這里還要強(qiáng)調(diào)一點(diǎn)的就是,對于訪問控制,應(yīng)該在程序里明確的指出,而不是依靠默認(rèn),這是一個良好的習(xí)慣,也讓你的代碼更具可讀性。?
說到這里,很多了解的人或許都認(rèn)為這個話題可以結(jié)束了,因?yàn)樗麄冎?span style="font-family:'Times New Roman'">struct和class的“唯一”區(qū)別就是訪問控制。很多文獻(xiàn)上也確實(shí)只提到這一個區(qū)別。?
但我上面卻沒有用“唯一”,而是說的“最本質(zhì)”,那是因?yàn)?#xff0c;它們確實(shí)還有另一個區(qū)別,雖然那個區(qū)別我們平時可能很少涉及。
那就是:“class”這個關(guān)鍵字還用于定義模板參數(shù),就像“typename”。但關(guān)鍵字“struct”不用于定義模板參數(shù)。這一點(diǎn)在Stanley?B.Lippman寫的Inside?the?C++?Object?Model有過說明。?
問題討論到這里,基本上應(yīng)該可以結(jié)束了。但有人曾說過,他還發(fā)現(xiàn)過其他的“區(qū)別”,那么,讓我們來看看,這到底是不是又一個區(qū)別。還是上面所說?的,C++中的struct是對C中的struct的擴(kuò)充,既然是擴(kuò)充,那么它就要兼容過去C中struct應(yīng)有的所有特性。例如你可以這樣寫:?
struct?A?//定義一個struct
{
???char?c1;
???int?n2;
???double?db3;
};
A?a={'p',?7,?3.1415926};?//定義時直接賦值?
也就是說struct可以在定義的時候用{}賦初值。那么問題來了,class行不行呢?將上面的struct改成class,試試看。報錯!噢~于是那人跳出來說,他又找到了一個區(qū)別。我們仔細(xì)看看,這真的又是一個區(qū)別嗎??
你試著向上面的struct中加入一個構(gòu)造函數(shù)(或虛函數(shù)),你會發(fā)現(xiàn)什么?
對,struct也不能用{}賦初值了
的確,以{}的方式來賦初值,只是用一個初始化列表來對數(shù)據(jù)進(jìn)行按順序的初始化,如上面如果寫成A?a={'p',7};則c1,n2被初始化,而db3沒有。這樣簡單的copy操作,只能發(fā)生在簡單的數(shù)據(jù)結(jié)構(gòu)上,而不應(yīng)該放在對象上。加入一個構(gòu)造函數(shù)或是一個虛函數(shù)會使struct更體現(xiàn)出一種對象的特性,而使此{}操作不再有效。?
事實(shí)上,是因?yàn)榧尤脒@樣的函數(shù),使得類的內(nèi)部結(jié)構(gòu)發(fā)生了變化。而加入一個普通的成員函數(shù)呢?你會發(fā)現(xiàn){}依舊可用。其實(shí)你可以將普通的函數(shù)理解成對數(shù)據(jù)結(jié)構(gòu)的一種算法,這并不打破它數(shù)據(jù)結(jié)構(gòu)的特性。?
那么,看到這里,我們發(fā)現(xiàn)即使是struct想用{}來賦初值,它也必須滿足很多的約束條件,這些條件實(shí)際上就是讓struct更體現(xiàn)出一種數(shù)據(jù)機(jī)構(gòu)而不是類的特性。?
那為什么我們在上面僅僅將struct改成class,{}就不能用了呢?
其實(shí)問題恰巧是我們之前所講的——訪問控制!你看看,我們忘記了什么?對,將struct改成class的時候,訪問控制由public變?yōu)?private了,那當(dāng)然就不能用{}來賦初值了。加上一個public,你會發(fā)現(xiàn),class也是能用{}的,和struct毫無區(qū)別!!!?
做個總結(jié),從上面的區(qū)別,我們可以看出,struct更適合看成是一個數(shù)據(jù)結(jié)構(gòu)的實(shí)現(xiàn)體,class更適合看成是一個對象的實(shí)現(xiàn)體。
總結(jié)
以上是生活随笔為你收集整理的C++中Struct和Class的区别的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 用Python生成随机的中文验证码图片
- 下一篇: 三条总线作用