当前位置: 首页 > 编程笔记 >

详解C++编程中的私有继承和公有继承

慕容星晖
2023-03-14
本文向大家介绍详解C++编程中的私有继承和公有继承,包括了详解C++编程中的私有继承和公有继承的使用技巧和注意事项,需要的朋友参考一下

C++类的私有继承
在声明一个派生类时将基类的继承方式指定为private的,称为私有继承,用私有继承方式建立的派生类称为私有派生类(private derived class ), 其基类称为私有基类(private base class )。

私有基类的公用成员和保护成员在派生类中的访问属性相当于派生类中的私有成员,即派生类的成员函数能访问它们,而在派生类外不能访问它们。私有基类的私有成员在派生类中成为不可访问的成员,只有基类的成员函数可以引用它们。一个基类成员在基类中的访问属性和在派生类中的访问属性可能是不同的。私有基类的成员在私有派生类中的访问属性见下表。

上表不必死记硬背,只需理解:既然声明为私有继承,就表示将原来能被外界引用的成员隐藏起来,不让外界引用,因此私有基类的公用成员和保护成员理所当然地成为派生类中的私有成员。

私有基类的私有成员按规定只能被基类的成员函数引用,在基类外当然不能访问他们,因此它们在派生类中是隐蔽的,不可访问的。

对于不需要再往下继承的类的功能可以用私有继承方式把它隐蔽起来,这样,下一层的派生类无法访问它的任何成员。可以知道,一个成员在不同的派生层次中的访问属性可能是不同的,它与继承方式有关。

[例]

class Student1: private Student//用私有继承方式声明派生类Student1
{
public:
  void display_1( ) //输出两个数据成员的值
  {
   cout<<"age: "<<age<<endl; //引用派生类的私有成员,正确
   cout<<"address: "<<addr<<endl;
  } //引用派生类的私有成员,正确
private:
  int age;
  string addr;
};

请分析下面的主函数:

int main( )
{
  Student1 stud1;//定义一个Student1类的对象stud1
  stud1.display(); //错误,私有基类的公用成员函数在派生类中是私有函数
  stud1.display_1( );//正确,Display_1函数是Student1类的公用函数
  stud1.age=18; //错误,外界不能引用派生类的私有成员
  return 0;
}

可以看到:
不能通过派生类对象(如stud1)引用从私有基类继承过来的任何成员(如stud1.display()或stud1.num)。
派生类的成员函数不能访问私有基类的私有成员,但可以访问私有基类的公用成员(如stud1.display_1函数可以调用基类的公用成员函数display,但不能引用基类的私有成员num)。

不少读者提出这样一个问題:私有基类的私有成员mun等数据成员只能被基类的成员函数引用,而私有基类的公用成员函数又不能被派生类外调用,那么,有没有办法调用私有基类的公用成员函数,从而引用私有基类的私有成员呢?有。

应当注意到,虽然在派生类外不能通过派生类对象调用私有基类的公用成员函数,但可以通过派生类的成员函数调用私有基类的公用成员函数(此时它是派生类中的私有成员函数,可以被派生类的任何成员函数调用)。

可将上面的私有派生类的成员函数定义改写为:

void display_1( )//输出5个数据成员的值
{
  display(): //调用基类的公用成员函数,输出3个数据成员的值
  cout<<"age: "<<age<<endl; //输出派生类的私有数据成员
  cout<<"address: "<<addr<<endl;
} //输出派生类的私有数据成员

main函数可改写为:

int main( )
{
  Student1 stud1;
  stud1.display_1( );//display_1函数是派生类Student1类的公用函数
  return 0;
}

这样就能正确地引用私有基类的私有成员。可以看到,本例采用的方法是:
在main函数中调用派生类中的公用成员函数stud1.display_1;
通过该公用成员函数调用基类的公用成员函数display(它在派生类中是私有函数,可以被派生类中的任何成员函数调用);
通过基类的公用成员函数display引用基类中的数据成员。

请根据上面的要求,补充和完善上面的程序,使之成为完整、正确的程序,程序中应包括输入数据的函数。

由于私有派生类限制太多,使用不方便,一般不常使用。

C++类的公用继承
在定义一个派生类时将基类的继承方式指定为public的,称为公用继承,用公用继承方式建立的派生类称为公用派生类(public derived class ),其基类称为公用基类(public base class )。

采用公用继承方式时,基类的公用成员和保护成员在派生类中仍然保持其公用成员和保护成员的属性,而基类的私有成员在派生类中并没有成为派生类的私有成员,它仍然是基类的私有成员,只有基类的成员函数可以引用它,而不能被派生类的成员函数引用,因此就成为派生类中的不可访问的成员。公用基类的成员在派生类中的访问属性见表。

有人问,既然是公用继承,为什么不让访问基类的私有成员呢?要知道,这是C++中一个重要的软件工程观点。因为私有成员体现了数据的封装性,隐藏私有成员有利于测试、调试和修改系统。如果把基类所有成员的访问权限都原封不动地继承到派生类,使基类的私有成员在派生类中仍保持其私有性质,派生类成员能访问基类的私有成员,那么岂非基类和派生类没有界限了?这就破坏了基类的封装性。如果派生类再继续派生一个新的派生类,也能访问基类的私有成员,那么在这个基类的所有派生类的层次上都能访问基类的私有成员,这就完全丢弃了封装性带来的好处。保护私有成员是一条重要的原则。

[例] 访问公有基类的成员。下面写出类的声明部分:

Class Student//声明基类
{
public: //基类公用成员
  void get_value( )
  {
   cin>>num>>name>>sex;
  }
  void display( )
  {
   cout<<" num: "<<num<<endl;
   cout<<" name: "<<name<<endl;
   cout<<" sex: "<<sex<<endl;
  }
private: //基类私有成员
  int num;
  string name;
  char sex;
};
class Student1: public Student //以public方式声明派生类Student1
{
public:
  void display_1( )
  {
   cout<<" num: "<<num<<endl; //企图引用基类的私有成员,错误
   cout<<" name: "<<name<<endl; //企图引用基类的私有成员,错误
   cout<<" sex: "<<sex<<endl; //企图引用基类的私有成员,错误
   cout<<" age: "<<age<<endl; //引用派生类的私有成员,正确
   cout<<" address: "<<addr<<endl;
  } //引用派生类的私有成员,正确
private:
  int age;
  string addr;
};

由于基类的私有成员对派生类来说是不可访问的,因此在派生类中的display_1函数中直接引用基类的私有数据成员num,name和sex是不允许的。只能通过基类的公用成员函数来引用基类的私有数据成员。可以将派生类Student1的声明改为

class Student1: public Student //以public方式声明派生类Student1
{
public:
  void display_1( )
  {
   cout<<" age: "<<age<<endl; //引用派生类的私有成员,正确
   cout<<" address: "<<addr<<endl; //引用派生类的私有成员,正确
  }
private:
  int age; string addr;
};

然后在main函数中分别调用基类的display函数和派生类中的display_1函数,先后输出5个数据。

可以这样写main函数(假设对象stud中已有数据):

int main( )
{
  Student1 stud;//定义派生类Student1的html" target="_blank">对象stud
  stud.display( ); //调用基类的公用成员函数,输出基类中3个数据成员的值
  stud.display_1(); //调用派生类公用成员函数,输出派生类中两个数据成员的值
  return 0;
}

请根据上面的分析,写出完整的程序,程序中应包括输入数据的函数。

实际上,程序还可以改进,在派生类的display_1函数中调用基类的display函数,在主函数中只要写一行:

  stud.display_1();


即可输出5个数据。

 类似资料:
  • 本文向大家介绍C/C++ 公有继承、保护继承和私有继承的对比详解,包括了C/C++ 公有继承、保护继承和私有继承的对比详解的使用技巧和注意事项,需要的朋友参考一下 C/C++ 公有继承、保护继承和私有继承的区别    在c++的继承控制中,有三种不同的控制权限,分别是public、protected和private。定义派生类时,若不显示加上这三个关键字,就会使用默认的方式,用struct定义的类

  • FAQs in section [24]: [24.1] 如何表示“私有继承”? [24.2] 私有继承和组合(composition)有什么类似? [24.3] 我应该选谁:组合还是私有继承? [24.4] 从私有继承类到父类需要指针类型转换吗? [24.5] 保护继承和私有继承的关系是什么? [24.6] 私有继承和保护继承的访问规则是什么? 24.1 如何表示“私有继承”? 用 : priv

  • 本文向大家介绍C++ 中私有继承的作用,包括了C++ 中私有继承的作用的使用技巧和注意事项,需要的朋友参考一下 C++ 中私有继承的作用 私有继承的 第一个规则:和公有继承相反,如果两个类之间的继承关系为私有,编译器一般不会将派生类对象转换成基类对象。 第二个规则: 从私有基类继承而来的成员都成为了派生类的私有成员,即使它们在基类中是保护或公有成员。 私有继承的含义:私有继承意味着 "用...来实

  • 问题内容: 这是一个有趣的代码片段: 我知道子类不会继承其父级的私有成员,但是obj设法在其中调用它不应具有的访问权的方法。在编译时,obj是Superclass类型,在运行时是Subclass类型。 这可能与以下事实有关:对doSomething()的调用发生在驱动程序类内部,而该驱动程序类恰好是其自己的类(以及为什么可以首先调用doSomething()的原因)。 因此问题归结为,obj如何访

  • 问题内容: 有人告诉我,对于Java子类,它可以继承其超类的所有成员。那么这是否意味着甚至私人成员?我知道它可以继承受保护的成员。 谁可以给我解释一下这个。我现在很困惑。 问题答案: 不,私有成员 不会被继承, 因为私有成员的范围 仅限 于定义它的类。仅公共成员和受保护成员被继承。 从, 超类私人成员 子类 不继承 其父类 的私有成员 。但是,如果超类具有用于访问其私有字段的公共或受保护的方法,则

  • 问题内容: 我对在继承中使用私有方法感到困惑,例如: 基于上面的代码,我感到困惑的私有方法的继承,是继承了私有方法来?还是两个类中的say方法完全无关?由于代码在main()方法中运行时出错,因此似乎无法从中调用私有方法。 问题答案: 如果您想让子类访问需要保留的超类方法,那么您正在寻找的关键字。 只允许包含该成员的类访问该成员。 允许在类及其所有子类中访问成员。 允许任何人访问该成员。