深入C++中struct與class的區(qū)別分析
更新時(shí)間:2013年05月28日 12:01:58 作者:
本篇文章是對(duì)C++中struct與class的區(qū)別進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
一、
C++中的struct對(duì)C中的struct進(jìn)行了擴(kuò)充,它已經(jīng)不再只是一個(gè)包含不同數(shù)據(jù)類型的數(shù)據(jù)結(jié)構(gòu)了,它已經(jīng)獲取了太多的功能。
struct能包含成員函數(shù)嗎? 能!
struct能繼承嗎? 能?。?BR>struct能實(shí)現(xiàn)多態(tài)嗎? 能!??!
最本質(zhì)的一個(gè)區(qū)別就是默認(rèn)的訪問控制,體現(xiàn)在兩個(gè)方面:
1)默認(rèn)的繼承訪問權(quán)限。struct是public的,class是private的。
寫如下的代碼:
struct A
{
char a;
};
struct B : A
{
char b;
};
這個(gè)時(shí)候B是public繼承A的。如果都將上面的struct改成class,那么B是private繼承A的。這就是默認(rèn)的繼承訪問權(quán)限。所以我們?cè)谄綍r(shí)寫類繼承的時(shí)候,通常會(huì)這樣寫:
struct 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繼承
2)struct作為數(shù)據(jù)結(jié)構(gòu)的實(shí)現(xiàn)體,它默認(rèn)的數(shù)據(jù)訪問控制是public的,而class作為對(duì)象的實(shí)現(xiàn)體,它默認(rèn)的成員變量訪問控制是private的。
3)“class”這個(gè)關(guān)鍵字還用于定義模板參數(shù),就像“typename”。但關(guān)鍵字“struct”不用于定義模板參數(shù)。
4) 還是上面所說的,C++中的struct是對(duì)C中的struct的擴(kuò)充,既然是擴(kuò)充,那么它就要兼容過去C中struct應(yīng)有的所有特性。例如你可以這樣寫:
struct A//定義一個(gè)struct
{
char c1;
int n2;
double db3;
};
A a={'p',7,3.1415926}; //定義時(shí)直接賦值
也就是說struct可以在定義的時(shí)候用{}賦初值。
向上面的struct中加入一個(gè)構(gòu)造函數(shù)(或虛函數(shù)),struct也不能用{}賦初值了。的確,以{}的方式來賦初值,只是用一個(gè)初始化列表來對(duì)數(shù)據(jù)進(jìn)行按順序的初始化,如上面如果寫成A a={'p',7};則c1,n2被初始化,而db3沒有。這樣簡(jiǎn)單的copy操作,只能發(fā)生在簡(jiǎn)單的數(shù)據(jù)結(jié)構(gòu)上,而不應(yīng)該放在對(duì)象上。加入一個(gè)構(gòu)造函數(shù)或是一個(gè)虛函數(shù)會(huì)使struct更體現(xiàn)出一種對(duì)象的特性,而使此{(lán)}操作不再有效。事實(shí)上,是因?yàn)榧尤脒@樣的函數(shù),使得類的內(nèi)部結(jié)構(gòu)發(fā)生了變化。而加入一個(gè)普通的成員函數(shù)呢?你會(huì)發(fā)現(xiàn){}依舊可用。其實(shí)你可以將普通的函數(shù)理解成對(duì)數(shù)據(jù)結(jié)構(gòu)的一種算法,這并不打破它數(shù)據(jù)結(jié)構(gòu)的特性。至于虛函數(shù)和普通成員函數(shù)有什么區(qū)別,我會(huì)具體寫篇文章討論。
那么,看到這里,我們發(fā)現(xiàn)即使是struct想用{}來賦初值,它也必須滿足很多的約束條件,這些條件實(shí)際上就是讓struct更體現(xiàn)出一種數(shù)據(jù)機(jī)構(gòu)而不是類的特性。那為什么我們?cè)谏厦鎯H僅將struct改成class,{}就不能用了呢?其實(shí)問題恰巧是我們之前所講的——訪問控制!你看看,我們忘記了什么?對(duì),將struct改成class的時(shí)候,訪問控制由public變?yōu)閜rivate了,那當(dāng)然就不能用{}來賦初值了。加上一個(gè)public,你會(huì)發(fā)現(xiàn),class也是能用{}的,和struct毫無區(qū)別?。?!
從上面的區(qū)別,我們可以看出,struct更適合看成是一個(gè)數(shù)據(jù)結(jié)構(gòu)的實(shí)現(xiàn)體,class更適合看成是一個(gè)對(duì)象的實(shí)現(xiàn)體。
二、
關(guān)于使用大括號(hào)初始化
class和struct如果定義了構(gòu)造函數(shù)的話,都不能用大括號(hào)進(jìn)行初始化
如果沒有定義構(gòu)造函數(shù),struct可以用大括號(hào)初始化。
如果沒有定義構(gòu)造函數(shù),且所有成員變量全是public的話,可以用大括號(hào)初始化。
關(guān)于默認(rèn)訪問權(quán)限
class中默認(rèn)的成員訪問權(quán)限是private的,而struct中則是public的。
關(guān)于繼承方式
class繼承默認(rèn)是private繼承,而struct繼承默認(rèn)是public繼承。
且看如下代碼(看看編譯器給出的錯(cuò)誤信息):
class T1
{
public:
void f()
{
cout<<"T1::f()"<<endl;
}
int x,y;
};
struct T2
{
int x;
void f(){cout<<"T2::f()"<<endl;}
};
struct TT1 : T1
{
};
class TT2 : T2
{
};
int main()
{
TT1 t1;
TT2 t2;
t1.f();
t2.f();
}
關(guān)于模版
在模版中,類型參數(shù)前面可以使用class或typename,如果使用struct,則含義不同,struct后面跟的是“non-type template parameter”,而class或typename后面跟的是類型參數(shù)。
template <struct X>
void f(X x)
{
}
//出錯(cuò)信息:d:codecpptestcpptestcpptest.cpp(33) : error C2065: 'X' : undeclared identifier
C++中的struct對(duì)C中的struct進(jìn)行了擴(kuò)充,它已經(jīng)不再只是一個(gè)包含不同數(shù)據(jù)類型的數(shù)據(jù)結(jié)構(gòu)了,它已經(jīng)獲取了太多的功能。
struct能包含成員函數(shù)嗎? 能!
struct能繼承嗎? 能?。?BR>struct能實(shí)現(xiàn)多態(tài)嗎? 能!??!
最本質(zhì)的一個(gè)區(qū)別就是默認(rèn)的訪問控制,體現(xiàn)在兩個(gè)方面:
1)默認(rèn)的繼承訪問權(quán)限。struct是public的,class是private的。
寫如下的代碼:
復(fù)制代碼 代碼如下:
struct A
{
char a;
};
struct B : A
{
char b;
};
這個(gè)時(shí)候B是public繼承A的。如果都將上面的struct改成class,那么B是private繼承A的。這就是默認(rèn)的繼承訪問權(quán)限。所以我們?cè)谄綍r(shí)寫類繼承的時(shí)候,通常會(huì)這樣寫:
struct 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繼承
2)struct作為數(shù)據(jù)結(jié)構(gòu)的實(shí)現(xiàn)體,它默認(rèn)的數(shù)據(jù)訪問控制是public的,而class作為對(duì)象的實(shí)現(xiàn)體,它默認(rèn)的成員變量訪問控制是private的。
3)“class”這個(gè)關(guān)鍵字還用于定義模板參數(shù),就像“typename”。但關(guān)鍵字“struct”不用于定義模板參數(shù)。
4) 還是上面所說的,C++中的struct是對(duì)C中的struct的擴(kuò)充,既然是擴(kuò)充,那么它就要兼容過去C中struct應(yīng)有的所有特性。例如你可以這樣寫:
struct A//定義一個(gè)struct
{
char c1;
int n2;
double db3;
};
A a={'p',7,3.1415926}; //定義時(shí)直接賦值
也就是說struct可以在定義的時(shí)候用{}賦初值。
向上面的struct中加入一個(gè)構(gòu)造函數(shù)(或虛函數(shù)),struct也不能用{}賦初值了。的確,以{}的方式來賦初值,只是用一個(gè)初始化列表來對(duì)數(shù)據(jù)進(jìn)行按順序的初始化,如上面如果寫成A a={'p',7};則c1,n2被初始化,而db3沒有。這樣簡(jiǎn)單的copy操作,只能發(fā)生在簡(jiǎn)單的數(shù)據(jù)結(jié)構(gòu)上,而不應(yīng)該放在對(duì)象上。加入一個(gè)構(gòu)造函數(shù)或是一個(gè)虛函數(shù)會(huì)使struct更體現(xiàn)出一種對(duì)象的特性,而使此{(lán)}操作不再有效。事實(shí)上,是因?yàn)榧尤脒@樣的函數(shù),使得類的內(nèi)部結(jié)構(gòu)發(fā)生了變化。而加入一個(gè)普通的成員函數(shù)呢?你會(huì)發(fā)現(xiàn){}依舊可用。其實(shí)你可以將普通的函數(shù)理解成對(duì)數(shù)據(jù)結(jié)構(gòu)的一種算法,這并不打破它數(shù)據(jù)結(jié)構(gòu)的特性。至于虛函數(shù)和普通成員函數(shù)有什么區(qū)別,我會(huì)具體寫篇文章討論。
那么,看到這里,我們發(fā)現(xiàn)即使是struct想用{}來賦初值,它也必須滿足很多的約束條件,這些條件實(shí)際上就是讓struct更體現(xiàn)出一種數(shù)據(jù)機(jī)構(gòu)而不是類的特性。那為什么我們?cè)谏厦鎯H僅將struct改成class,{}就不能用了呢?其實(shí)問題恰巧是我們之前所講的——訪問控制!你看看,我們忘記了什么?對(duì),將struct改成class的時(shí)候,訪問控制由public變?yōu)閜rivate了,那當(dāng)然就不能用{}來賦初值了。加上一個(gè)public,你會(huì)發(fā)現(xiàn),class也是能用{}的,和struct毫無區(qū)別?。?!
從上面的區(qū)別,我們可以看出,struct更適合看成是一個(gè)數(shù)據(jù)結(jié)構(gòu)的實(shí)現(xiàn)體,class更適合看成是一個(gè)對(duì)象的實(shí)現(xiàn)體。
二、
關(guān)于使用大括號(hào)初始化
class和struct如果定義了構(gòu)造函數(shù)的話,都不能用大括號(hào)進(jìn)行初始化
如果沒有定義構(gòu)造函數(shù),struct可以用大括號(hào)初始化。
如果沒有定義構(gòu)造函數(shù),且所有成員變量全是public的話,可以用大括號(hào)初始化。
關(guān)于默認(rèn)訪問權(quán)限
class中默認(rèn)的成員訪問權(quán)限是private的,而struct中則是public的。
關(guān)于繼承方式
class繼承默認(rèn)是private繼承,而struct繼承默認(rèn)是public繼承。
且看如下代碼(看看編譯器給出的錯(cuò)誤信息):
復(fù)制代碼 代碼如下:
class T1
{
public:
void f()
{
cout<<"T1::f()"<<endl;
}
int x,y;
};
struct T2
{
int x;
void f(){cout<<"T2::f()"<<endl;}
};
struct TT1 : T1
{
};
class TT2 : T2
{
};
int main()
{
TT1 t1;
TT2 t2;
t1.f();
t2.f();
}
關(guān)于模版
在模版中,類型參數(shù)前面可以使用class或typename,如果使用struct,則含義不同,struct后面跟的是“non-type template parameter”,而class或typename后面跟的是類型參數(shù)。
template <struct X>
void f(X x)
{
}
//出錯(cuò)信息:d:codecpptestcpptestcpptest.cpp(33) : error C2065: 'X' : undeclared identifier
您可能感興趣的文章:
- C++中的struct和class的區(qū)別詳解
- C++深入探索類和對(duì)象之封裝及class與struct的區(qū)別
- C++詳細(xì)講解函數(shù)調(diào)用與Struct和CLass的區(qū)別
- C++深入探索類真正的形態(tài)之struct與class
- C++基礎(chǔ) class、struct、union詳細(xì)
- C++ class和struct到底有什么區(qū)別詳解
- C++結(jié)構(gòu)體struct和類class區(qū)別詳解
- C++中聲明類的class與聲明結(jié)構(gòu)體的struct關(guān)鍵字詳解
- C++中關(guān)鍵字Struct和Class的區(qū)別
- c++中struct和class的區(qū)別小結(jié)
相關(guān)文章
C語言數(shù)據(jù)結(jié)構(gòu)遞歸之斐波那契數(shù)列
這篇文章主要介紹了C語言數(shù)據(jù)結(jié)構(gòu)遞歸之斐波那契數(shù)列的相關(guān)資料,希望通過本文能幫助到大家,讓大家理解掌握這部分內(nèi)容,需要的朋友可以參考下2017-10-10C++實(shí)現(xiàn)拷貝構(gòu)造函數(shù)的方法詳解
拷貝構(gòu)造函數(shù)是構(gòu)造函數(shù)的一個(gè)重載,因此顯式的定義了拷貝構(gòu)造,那么編譯器也不再默認(rèn)生成構(gòu)造函數(shù)。本文主要介紹了C++實(shí)現(xiàn)拷貝構(gòu)造函數(shù)的方法,需要的可以參考一下2022-09-09C++實(shí)現(xiàn)模板中的非類型參數(shù)的方法
這篇文章主要介紹了C++實(shí)現(xiàn)模板中的非類型參數(shù)的方法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-03-03c++報(bào)錯(cuò)問題解決方案lvalue required as left opera
這篇文章主要介紹了c++報(bào)錯(cuò):lvalue required as left operand of assignment,出現(xiàn)此錯(cuò)誤原因,是因?yàn)?,等?hào)左邊是不可被修改的表達(dá)式或常量,而表達(dá)式或常量不能作為左值,需要的朋友可以參考下2023-01-01利用C++實(shí)現(xiàn)計(jì)算機(jī)輔助教學(xué)系統(tǒng)
我們都知道計(jì)算機(jī)在教育中起的作用越來越大。這篇文章主要為大家詳細(xì)介紹了如何利用C++編寫一個(gè)計(jì)算機(jī)輔助教學(xué)系統(tǒng),感興趣的可以了解一下2023-05-05C++中new與delete、malloc與free應(yīng)用分析
這篇文章主要介紹了C++中new與delete、malloc與free應(yīng)用分析,很重要的概念,需要的朋友可以參考下2014-08-08