C++ 繼承公有、私有、保護
公有繼承(public)、私有繼承(private)、保護繼承(protected)是常用的三種繼承方式。
1. 公有繼承(public)
公有繼承的特點是基類的公有成員和保護成員作為派生類的成員時,它們都保持原有的狀態,而基類的私有成員仍然是私有的,不能被這個派生類的子類所訪問。
2. 私有繼承(private)
私有繼承的特點是基類的公有成員和保護成員都作為派生類的私有成員,並且不能被這個派生類的子類所訪問。
3. 保護繼承(protected)
保護繼承的特點是基類的所有公有成員和保護成員都成為派生類的保護成員,並且只能被它的派生類成員函式或友元訪問,基類的私有成員仍然是私有的。
下面列出三種不同的繼承方式的基類特性和派生類特性。
publicprotectedprivate
共有繼承public protected不可見
私有繼承private private不可見
保護繼承protectedprotected不可見
在上圖中:1)基類成員對派生類都是:共有和保護的成員是可見的,私有的的成員是不可見的。
2)基類成員對派生類的物件來說:要看基類的成員在派生類中變成了什麼型別的成員。如:私有繼承時,基類的共有成員和私有成員都變成了派生類中的私有成員,因此對於派生類中的物件來說基類的共有成員和私有成員就是不可見的。
為了進一步理解三種不同的繼承方式在其成員的可見性方面的區別,下面從三種不同角度進行討論。
對於公有繼承方式
(1) 基類成員對其物件的可見性:
公有成員可見,其他不可見。這裡保護成員同於私有成員。
(2) 基類成員對派生類的可見性:
公有成員和保護成員可見,而私有成員不可見。這裡保護成員同於公有成員。
(3) 基類成員對派生類物件的可見性:
公有成員可見,其他成員不可見。
所以,在公有繼承時,派生類的物件可以訪問基類中的公有成員;派生類的成員函式可以訪問基類中的公有成員和保護成員。這裡,一定要區分清楚派生類的物件和派生類中的成員函式對基類的訪問是不同的。
對於私有繼承方式
(1) 基類成員對其物件的可見性:
公有成員可見,其他成員不可見。
(2) 基類成員對派生類的可見性:
公有成員和保護成員是可見的,而私有成員是不可見的。
(3) 基類成員對派生類物件的可見性:
所有成員都是不可見的。
所以,在私有繼承時,基類的成員只能由直接派生類訪問,而無法再往下繼承。
對於保護繼承方式
這種繼承方式與私有繼承方式的情況相同。兩者的區別僅在於對派生類的成員而言,對基類成員有不同的可見性。
上述所說的可見性也就是可訪問性。
關於可訪問性還有另的一種說法。這種規則中,稱派生類的物件對基類訪問為水平訪問,稱派生類的派生類對基類的訪問為垂直訪問。
看看這樣的例子:
#include<iostream>
using namespace std;
//////////////////////////////////////////////////////////////////////////
class A //父類
{
private:
int privatedateA;
protected:
int protecteddateA;
public:
int publicdateA;
};
//////////////////////////////////////////////////////////////////////////
class B :public A //基類A的派生類B(共有繼承)
{
public:
void funct()
{
int b;
b=privatedateA; //error:基類中私有成員在派生類中是不可見的
b=protecteddateA; //ok:基類的保護成員在派生類中為保護成員
b=publicdateA; //ok:基類的公共成員在派生類中為公共成員
}
};
//////////////////////////////////////////////////////////////////////////
class C :private A //基類A的派生類C(私有繼承)
{
public:
void funct()
{
int c;
c=privatedateA; //error:基類中私有成員在派生類中是不可見的
c=protecteddateA; //ok:基類的保護成員在派生類中為私有成員
c=publicdateA; //ok:基類的公共成員在派生類中為私有成員
}
};
//////////////////////////////////////////////////////////////////////////
class D :protected A //基類A的派生類D(保護繼承)
{
public:
void funct()
{
int d;
d=privatedateA; //error:基類中私有成員在派生類中是不可見的
d=protecteddateA; //ok:基類的保護成員在派生類中為保護成員
d=publicdateA; //ok:基類的公共成員在派生類中為保護成員
}
};
//////////////////////////////////////////////////////////////////////////
int main()
{
int a;
B objB;
a=objB.privatedateA; //error:基類中私有成員在派生類中是不可見的,對物件不可見
a=objB.protecteddateA; //error:基類的保護成員在派生類中為保護成員,對物件不可見
a=objB.publicdateA; //ok:基類的公共成員在派生類中為公共成員,對物件可見
C objC;
a=objC.privatedateA; //error:基類中私有成員在派生類中是不可見的,對物件不可見
a=objC.protecteddateA; //error:基類的保護成員在派生類中為私有成員,對物件不可見
a=objC.publicdateA; //error:基類的公共成員在派生類中為私有成員,對物件不可見
D objD;
a=objD.privatedateA; //error:基類中私有成員在派生類中是不可見的,對物件不可見
a=objD.protecteddateA; //error:基類的保護成員在派生類中為保護成員,對物件不可見
a=objD.publicdateA; //error:基類的公共成員在派生類中為保護成員,對物件不可見
return 0;
}