C++继承:公有,私有,保护

本文详细解析了C++中公有继承、私有继承和保护继承的区别与特点,包括基类成员在派生类中的可见性、可访问性以及对派生类对象的影响。通过实例代码演示了不同继承方式下基类成员如何在派生类中被访问,解释了公有继承、私有继承和保护继承的主要特征。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

公有继承(public)、私有继承(private)、保护继承(protected)是常用的三种继承方式。

1. 公有继承(public)

公有继承的特点是基类的公有成员和保护成员作为派生类的成员时,它们都保持原有的状态,而基类的私有成员仍然是私有的,不能被这个派生类的子类所访问。

2. 私有继承(private)

私有继承的特点是基类的公有成员和保护成员都作为派生类的私有成员,并且不能被这个派生类的子类所访问。

3. 保护继承(protected)

保护继承的特点是基类的所有公有成员和保护成员都成为派生类的保护成员,并且只能被它的派生类成员函数或友元访问,基类的私有成员仍然是私有的。

下面列出三种不同的继承方式的基类特性和派生类特性。

 publicprotectedprivate
共有继承publicprotected不可见
私有继承privateprivate不可见
保护继承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的派生类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;
}
======================================================================================================================================================

  1. /********** 
  2. Inheri.h 
  3. **********/ 
  4. #include <iostream>   
  5. using namespace std;  
  6. /********************************************************************* 
  7. *******公有派生:派生类对象包含基类对象,基类公有成员成为 
  8. 派生类的公有成员,私有成员也成为派生类的一部分,但只能通 
  9. 过基类的共有和保护方法访问。 
  10. **********************************************************************/  
  11. class A  
  12. {  
  13. public:  
  14.     A()  
  15.     {  
  16.         cout<<"A::A()"<<endl;  
  17.     }  
  18.     virtual ~A()//为何需要虚拟析构函数?虽然这好像并不需要。但是,这是必   
  19.                 //要的习惯。如Parent* p = new Child;delete p;   
  20.                 //它能够保证正确的析构序列,即先析构子类对象再析构基类对象   
  21.                 //(与构造函数相反),否则对象只析构基类对象,   
  22.                 //子类对象则没有释放   
  23.     {  
  24.         cout<<"A::~A()"<<endl;  
  25.     }  
  26. };  
  27. class B:public A  
  28. {  
  29. public:  
  30.     B():c(0)//公有继承的构造方法:不调用基类构造方法,程序将使用默认的基类构   
  31.             //造方法。显式调用基类构造方法可以初始化继承自基类的私有变量;   
  32.             //还可以使用成员初始化列表对成员变量进行初始化(如本例对c的初始化)   
  33.     {  
  34.         cout<<"B::B()"<<endl;  
  35.         fun();  
  36.     }  
  37.     virtual ~B()//为何需要虚拟析构函数?——同上   
  38.     {  
  39.         cout<<"B::~B()"<<endl;  
  40.     }  
  41.     virtual void fun()  
  42.     {  
  43.         cout<<"B::fun"<<endl;  
  44.     }  
  45.     virtual void test(int a=10)//虚函数有何特性?   
  46.     {  
  47.         cout<<"a= "<<a<<endl;  
  48.     }  
  49. private:  
  50.     char c;  
  51. };  
  52. class C:public B  
  53. {  
  54. public:  
  55.     C():i(0)//派生类构造函数:基类对象首先被创建;派生类应当将(成员初始化列表)   
  56.             //基类信息传递给基类构造函数;   
  57.             //派生类构造函数应初始化派生类新增的数据成员   
  58.     {  
  59.         cout<<"C::C()"<<endl;  
  60.     }  
  61.     ~C()  
  62.     {  
  63.         cout<<"C::~C()"<<endl;  
  64.     }  
  65.     void fun()  
  66.     {  
  67.         cout<<"C::fun"<<endl;  
  68.     }  
  69.     void test(int a=20)  
  70.     {  
  71.         cout<<"a= "<<a<<endl;  
  72.     }  
  73. private:  
  74.     int i;  
  75. };  
  76. /*** 
  77. 1.派生类对象可以使用基类的公有方法和保护方法 
  78. 2.基类指针可以在不进行显式类型转化的情况下指向派生类对象,基类指针只能调用基类方法 
  79. 3.基类引用可以在不进行显式类型转化的情况下引用派生类对象,基类引用只能调用基类方法 
  80. 同一方法在基类和在派生类表现的行为不同,其行为取决于调用该方法的对象,这种复杂的行 
  81. 为称为多态。 
  82. 1.在派生类中重新定义基类方法。 
  83. 2.基类方法声明virtual,虚方法,如果方法是通过引用或指针而不是对象调用的,它将确定使 
  84. 用哪一种方法: 
  85.   使用virtual,程序根据引用类型或指针类型指向的对象选择方法,类似C#中的覆盖; 
  86.   不使用virtual,程序根据引用类型或指针类型选择方法,类似C#中的隐藏; 
  87.   简言之,virtual 使得子类方法 Override 基类方法。 
  88. ****/  
  89. /********** 
  90. main.cpp 
  91. **********/ 
  92. #include "Inheri.h"   
  93. int main()  
  94. {  
  95.     cout<<"===Construct A==="<<endl;  
  96.     A *a=new C;  
  97.     cout<<"===Construct B==="<<endl;  
  98.     B *b=new C;  
  99.     cout<<"===Satrt Work==="<<endl;  
  100.     b->test();  
  101.     cout<<"Size of A is "<<sizeof(A)<<endl;  
  102.     //用类创建的对象的内存空间是如何分配的?这个也是刚刚看到的。内容好多,   
  103.     //参考http://blog.youkuaiyun.com/guogangj/archive/2008/01/11/2036785.aspx,慢慢看,耐心看。   
  104.     cout<<"Size of B is "<<sizeof(B)<<endl;  
  105.     cout<<"Size of C is "<<sizeof(C)<<endl;  
  106.       
  107.     cout<<"==Delete b=="<<endl;  
  108.     delete b;  
  109.     cout<<"==Delete a=="<<endl;  
  110.     delete a;  
  111.       
  112.     return 0;  
  113. }  
  114. /********************************************************************** 
  115. *******在编译过程中进行的联编(bind),称为静态联编(早期联编)。 
  116. *******编译器在运行时生成选择虚方法的代码,这些代码用来选择正确的虚方法, 
  117. 称为动态联编(晚期联编)。 
  118. 编译器对废墟方法使用静态联编;编译器对虚方法使用动态联编。非虚函数的效率 
  119. 比虚函数稍高,但不具备动态联编的能力。 
  120. 编译器处理虚函数的方法:为每个对象怎家一个隐藏成员,用以保存一个指向函数 
  121. 地址数组的指针。该数组称为虚函数表。虚函数表存储了为类对象进行声明的虚函数 
  122. 的地址,额外开销为一个字。 
  123. 构造函数不能是虚方法,无意义。 
  124. 为基类定义虚拟析构函数。 
  125. 友元不能使虚函数。 
  126. *********************************************************************/  


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值