一、public 方式继承1 R6 U3 G4 z5 z5 Z' v5 m3 |
基类成员对派生类的可见性对派生类来说,基类的公有成员和保护成员可见,基类的公有成员和保护成员作为派生类的成员时,它们都保持原有的状态;基类的私有成员不可见,基类的私有成员仍然是私有的,派生类不可访问基类中的私有成员。
$ `, R/ l/ W) W! q$ L0 S 基类成员对派生类对象的可见性对派生类对象来说,基类的公有成员是可见的,其他成员是不可见的。
) d- ?. i) x. G# R: i3 D, ^/ I 所以,在公有继承时,派生类的对象可以访问基类中的公有成员,派生类的成员函数可以访问基类中的公有成员和保护成员。
: l* W }* \. E4 d. }, ~ 简单来说,派生类能访问基类的 public, protected 成员,继承过来权限不变,派生类对象只能访问基类 public 成员。& O. [* l9 n: E- v- j* t
测试代码如下:; T. Z1 t5 v4 F# ?" S5 X5 a
<p>class A</p><p>{</p><p>private:</p><p> int m_data1;</p><p> void print1() { cout << "private print1" << endl; }</p><p>protected:</p><p> int m_data2;</p><p> void print2() { cout << "protected print2" << endl; }</p><p>public:</p><p> A(int x = 1, int y = 2, int z = 3) : m_data1(x), m_data2(y), m_data3(z) {}</p><p> int m_data3;</p><p> void print3() { cout << "protected print3" << endl; }</p><p>};</p><p>
</p><p>class B : public A</p><p>{</p><p>public:</p><p> void test_public() {</p><p> cout << m_data3 << endl;</p><p> print3();</p><p> }</p><p> void test_protected() {</p><p> cout << m_data2 << endl;</p><p> print2();</p><p> }</p><p> void test_private() {</p><p> // 下面两行编译不过,B 类内无法访问父类的私有成员</p><p> // cout << m_data1 << endl; </p><p> // print1();</p><p> }</p><p>};</p><p>
</p><p>int main(int argc, char const* argv[])</p><p>{</p><p> B b;</p><p> b.test_public();</p><p> b.test_protected();</p><p> b.test_private();</p><p> cout << b.m_data3 << endl;</p><p> // cout << b.m_data2 << endl; // 编译不过,子类对象无法访问父类 protected 的成员</p><p> // cout << b.m_data1 << endl; // 编译不过,子类对象无法访问父类 private 的成员</p><p> return 0;</p><p>}</p> 二、private 方式继承
' H: H: J. Z6 ]$ G% b% ?4 ?" d1 J 基类成员对其对象的可见性与一般类及其对象的可见性相同,公有成员可见,其他成员不可见。
$ k7 ?' [, N+ Q* d9 l 基类成员对派生类的可见性对派生类来说,基类的公有成员和保护成员是可见的,基类的公有成员和保护成员都作为派生类的私有成员,并且不能被这个派生类的子类所访问;基类的私有成员是不可见的,派生类不可访问基类中的私有成员。$ J- ~7 s& U: d# Z
基类成员对派生类对象的可见性对派生类对象来说,基类的所有成员都是不可见的。所以,在私有继承时,基类的成员只能由直接派生类访问,而无法再往下继承。
6 [! x: J. \- z1 n* t: t0 s 简单来说派生类可以访问基类的 public, protected 成员,继承过来之后变成自己私有的。 派生类的对象啥都不能访问。
* K9 F. |. c& i/ d9 `<p>class A</p><p>{</p><p>private:</p><p> int m_data1;</p><p> void print1() { cout << "private print1" << endl; }</p><p>protected:</p><p> int m_data2;</p><p> void print2() { cout << "protected print2" << endl; }</p><p>public:</p><p> A(int x = 1, int y = 2, int z = 3) : m_data1(x), m_data2(y), m_data3(z) {}</p><p> int m_data3;</p><p> void print3() { cout << "protected print3" << endl; }</p><p>};</p><p>
</p><p>class B : private A</p><p>{</p><p>public:</p><p> void test_public() {</p><p> cout << m_data3 << endl;</p><p> print3();</p><p> }</p><p> void test_protected() {</p><p> cout << m_data2 << endl;</p><p> print2();</p><p> }</p><p> void test_private() {</p><p> // 下面两行编译不过,B 类内无法访问父类的私有成员</p><p> // cout << m_data1 << endl; </p><p> // print1();</p><p> }</p><p>};</p><p>
</p><p>int main(int argc, char const* argv[])</p><p>{</p><p> B b;</p><p> b.test_public();</p><p> b.test_protected();</p><p> b.test_private();</p><p> // cout << b.m_data3 << endl; // // 编译不过,子类对象无法访问父类 public 的成员</p><p> // cout << b.m_data2 << endl; // 编译不过,子类对象无法访问父类 protected 的成员</p><p> // cout << b.m_data1 << endl; // 编译不过,子类对象无法访问父类 private 的成员</p><p> return 0;</p><p>}</p> 三、protected 方式继承
% k5 M0 N d1 I) D$ X. @ 基类成员对派生类的可见性对派生类来说,基类的公有成员和保护成员是可见的,基类的公有成员和保护成员都作为派生类的保护成员,并且不能被这个派生类的子类的对象所访问,但可以被派生类的子类所访问;基类的私有成员是不可见的,派生类不可访问基类中的私有成员。- Z" n" Z( X2 ~. J: y/ ?
基类成员对派生类对象的可见性对派生类对象来说,基类的所有成员都是不可见的。# M% c. F; X! Q* P& z- u: Q
简单来说:派生类可以访问基类的 public, protected,继承过来都变成了 protected,派生类对象啥都不能访问。
* U/ Q4 J5 H0 G1 U( e3 c5 g* X' F四、总结
5 m; b; l( A( h' B" ^ ^) H 对于这三种方式继承的派生类来说都能访问基类的 public, protected 成员;. |1 Z% w- U" j2 f
public 的方式继承到派生类,这些成员的权限和在基类里的权限保持一致 J8 A" T: e8 }5 }. ~% D
protected 方式继承到派生类,成员的权限都变为 protected;" a# V) @8 n0 {; L f
private 方式继承到派生类,成员的权限都变为 private;
$ i" W/ I0 P( E- I0 h9 v+ u& y 对于三种方式派生类的对象来说:只有 public 的方式继承后,派生来的对象只能访问基类的 public 成员,protected 和 private 方式继承,派生类的对象都不可以访问父类的成员。
* p5 e! b, G2 M0 K" c" ] 例: 请考虑标记为 A 到 J 的语句在编译时可能出现的情况。7 D5 ^3 j9 H) `2 N O# H) D
<p>#include<iostream></p><p>#include<cstdio></p><p>
</p><p>class Parent</p><p>{</p><p>public:</p><p> Parent(int var=-1) {</p><p> m_nPub = var;</p><p> m_nPtd = var;</p><p> m_bPrt = var;</p><p> }</p><p> int m_nPub;</p><p>protected:</p><p> int m_nPtd;</p><p>private:</p><p> int m_nPrt;</p><p>};</p><p>
</p><p>class Child1 : public Parent</p><p>{</p><p>public:</p><p> int GetPub() { return m_nPub; }</p><p> int GetPtd() { return m_nPtd; }</p><p> int GetPrt() { return m_nPrt; }</p><p> // A</p><p>};</p><p>
</p><p>class Child2 : protected Parent</p><p>{</p><p>public:</p><p> int GetPub() { return m_nPub; }</p><p> int GetPtd() { return m_nPtd; }</p><p> int GetPrt() { return m_nPrt; }</p><p> // B</p><p>};</p><p>
</p><p>class Child3 : private Parent</p><p>{</p><p>public:</p><p> int GetPub() { return m_nPub; }</p><p> int GetPtd() { return m_nPtd; }</p><p> int GetPrt() { return m_nPrt; }</p><p> // C</p><p>};</p><p>
</p><p>int main(int argc, char const *argv[])</p><p>{</p><p> Child1 cd1;</p><p> Child2 cd2;</p><p> Child3 cd3;</p><p>
</p><p> int nVar = 0;</p><p>
</p><p> // public inherited</p><p> cd1.m_nPub = nVar; // D</p><p> cd1.m_nPtd = nVar; // E</p><p> nVar = cd1.GetPtd(); // F</p><p>
</p><p> // protected inherited</p><p> cd2.m_nPub = nVar; // G</p><p> nVar = cd2.GetPtd(); // H</p><p>
</p><p> // private inherited</p><p> cd3.m_nPub = nVar; // I</p><p> nVar = cd3.GetPtd(); // J</p><p> return 0;</p><p>}</p>
- A, B, C 都错误,因为 m_nPrt 是父类的 private 变量,子类不能访问。
- D 正确。cdl 是公有继承,可以访问并改变父类的公有变量。
- E 错误。m_nPtd 是父类 Parent 的保护变量,不可以被公有继承的 cdl 访问, 更不可以被修改。 虽然 m_nPtd 是父类 Parent 的保护变量,经过公有继承后,m_nPtd 在子类中依然是 protected, 而子类的对象 cdl 是不能访问自身的 protected 成员,只能访问 public 成员。
- F 正确。派生类内可以访问父类的保护变量。
- G 错误。cd2 是保护继承的,派生类对象不能访问父类成员。
- H 正确。派生类内可以访问父类的保护变量。
- I 错误。cd2 是私有继承的,派生类对象不能访问父类成员。
- J 正确。派生类内可以访问父类的保护变量。5 H) u+ g6 W8 [3 e' k
3 Z! C+ j4 y" d( g+ S
8 O% P- Z! c4 d2 p- | E$ K( |% O |% W
|