我们知道,如果一个派生类有多个直接基类,而这些直接基类又有一个共同的基类,则在最终的派生类中会保留该间接共同基类数据成员的多份同名成员。在引用这些同名的成员时,必须在派生类对象名后增加直接基类名,以避免产生二义性,使其惟一地标识一个成员,如:
    c1.A::display( )

在一个类中保留间接共同基类的多份同名成员,虽然有时是有必要的,可以在不同的数据成员中分别存放不同的数据,也可以通过构造函数分别对它们进行初始化。但在大多数情况下,这种现象是人们不希望出现的。因为保留多份数据成员的拷贝,不仅占用较多的存储空间,还增加了访问这些成员时的困难,容易出错。而且在实际上,并不需要有多份拷贝。

C++提供虚基类(virtual base class)的方法,使得在继承间接共同基类时只保留一份成员。假设类D是类B和类C公用派生类,而类B和类C又是类A的派生类,如图11.21所示。 设类A有数据成员data和成员函数fun;派生类B和C分别从类A继承了data和fun,此外类B还增加了自己的数据成员data_b,类C增加了数据成员data_c。如果不用虚基类,就会在类D中保留了类A成员data的两份拷贝,分别表示为int B::data和int C::data。同样有两个同名的成员函数,表示为void B::fun()和void C::fun()。类B中增加的成员data_b和类C中增加的成员dat_c不同名,不必用类名限定。此外,类D还增加了自己的数据成员data_d和成员函数fun_d。


图 11.21

现在,将类A声明为虚基类,方法如下:

  1. class A //声明基类A
  2. {
  3. // 代码
  4. };
  5. class B: virtual public A //声明类B是类A的公用派生类,A是B的虚基类
  6. {
  7. // 代码
  8. };
  9. class C: virtual public A //声明类C是类A的公用派生类,A是C的虚基类
  10. {
  11. // 代码
  12. };

注意: 虚基类并不是在声明基类时声明的,而是在声明派生类时,指定继承方式时声明的。因为一个基类可以在生成一个派生类时作为虚基类,而在生成另一个派生类时不作为虚基类。

声明虚基类的一般形式为:
   class 派生类名: virtual 继承方式  基类名
即在声明派生类时,将关键字 virtual 加到相应的继承方式前面,经过这样的声明后,当基类通过多条派生路径被一个派生类继承时,该派生类只继承该基类一次,也就是说,基类成员只保留一次。

需要注意:为了保证虚基类在派生类中只继承一次,应当在该基类的所有直接派生类中声明为虚基类。否则仍然会出现对基类的多次继承。

如果在派生类B和C中将类A声明为虚基类,而在派生类D中没有将类A声明为虚基类,则在派生类E中,虽然从类B和C路径派生的部分只保留一份基类成员,但从类D路径派生的部分还保留一份基类成员。

虚基类的初始化

如果在虚基类中定义了带参数的构造函数,而且没有定义默认构造函数,则在其所有派生类(包括直接派生或间接派生的派生类)中,通过构造函数的初始化表对虚基类进行初始化。例如

  1. class A //定义基类A
  2. {
  3. A(int i){ } //基类构造函数,有一个参数};
  4. class B :virtual public A //A作为B的虚基类
  5. {
  6. B(int n):A(n){ } //B类构造函数,在初始化表中对虚基类初始化
  7. };
  8. class C :virtual public A //A作为C的虚基类
  9. {
  10. C(int n):A(n){ } //C类构造函数,在初始化表中对虚基类初始化
  11. };
  12. class D :public B,public C //类D的构造函数,在初始化表中对所有基类初始化
  13. {
  14. D(int n):A(n),B(n),C(n){ }
  15. };

注意: 在定义类D的构造函数时,与以往使用的方法有所不同。以往,在派生类的构造函数中只需负责对其直接基类初始化,再由其直接基类负责对间接基类初始化。现在,由于虚基类在派生类中只有一份数据成员,所以这份数据成员的初始化必须由派生类直接给出。如果不由最后的派生类直接对虚基类初始化,而由虚基类的直接派生类(如类B和类C)对虚基类初始化,就有可能由于在类B和类C的构造函数中对虚基类给出不同的初始化参数而产生矛盾。所以规定:在最后的派生类中不仅要负责对其直接基类进行初始化,还要负责对虚基类初始化。

有的读者会提出:类D的构造函数通过初始化表调了虚基类的构造函数A,而类B和类C的构造函数也通过初始化表调用了虚基类的构造函数A,这样虚基类的构造函数岂非被调用了3次?大家不必过虑,C++编译系统只执行最后的派生类对虚基类的构造函数的调用,而忽略虚基类的其他派生类(如类B和类C) 对虚基类的构造函数的调用,这就保证了虚基类的数据成员不会被多次初始化。

虚基类的简单应用举例

[例11.9] 在例11. 8(具体代码请查看:C++类的多重继承)的基础上,在Teacher类和Student类之上增加一个共同的基类Person。作为人员的一些基本数据都放在Person中,在 Teacher类和Student类中再增加一些必要的数据。可写出以下程序:

  1. #include <iostream>
  2. #include <string>
  3. using namespace std;
  4. //声明公共基类Person
  5. class Person
  6. {
  7. public:
  8. Person(string nam,char s,int a) //构造函数
  9. {
  10. name=nam;
  11. sex=s;
  12. age=a;
  13. }
  14. protected: //保护成员
  15. string name;
  16. char sex;
  17. int age;
  18. };
  19. //声明Person的直接派生类Teacher
  20. class Teacher:virtual public Person //声明Person为公用继承的虚基类
  21. {
  22. public:
  23. Teacher(string nam,char s,int a, string t):Person(nam,s,a)//构造函数
  24. {
  25. title=t;
  26. }
  27. protected: //保护成员
  28. string title; //职称
  29. };
  30. //声明Person的直接派生类Student
  31. class Student:virtual public Person //声明Person为公用继承的虚基类
  32. {
  33. public:
  34. Student(string nam,char s,int a,float sco) //构造函数
  35. :Person(nam,s,a),score(sco){ } //初始化表
  36. protected: //保护成员
  37. float score; //成绩
  38. };
  39. //声明多重继承的派生类Graduate
  40. class Graduate:public Teacher,public Student //Teacher和Student为直接基类
  41. {
  42. public:
  43. Graduate(string nam,char s,int a, string t,float sco,float w)//构造函数
  44. :Person(nam,s,a),Teacher(nam,s,a,t),Student(nam,s,a,sco),wage(w){}
  45. //初始化表
  46. void show( ) //输出研究生的有关数据
  47. {
  48. cout<<"name:"<<name<<endl;
  49. cout<<"age:"<<age<<endl;
  50. cout<<"sex:"<<sex<<endl;
  51. cout<<"score:"<<score<<endl;
  52. cout<<"title:"<<title<<endl;
  53. cout<<"wages:"<<wage<<endl;
  54. }
  55. private:
  56. float wage; //工资
  57. };
  58. //主函数
  59. int main( )
  60. {
  61. Graduate grad1("Wang-li",'f',24,"assistant",89.5,1234.5);
  62. grad1.show( );
  63. return 0;
  64. }

对程序的两点说明:
1) 请注意各类的构造函数的写法。在Person类中定义了包含3个形参的构造函数,用它对数据成员name、sex和age进行初始化。在Teacher和Student类的构造函数中,按规定要在初始化表中包含对基类的初始化,尽管对虚基类来说,编译系统不会由此调用基类的构造函数,但仍然应当按照派生类构造函数的统一格式书写。在最后派生类Graduate的构造函数中,既包括对虚基类构造函数的调用,也包括对其直接基类的初 始化。

2) 在Graduate类中,只保留一份基类的成员,因此可以用Graduate类中的show函数引用Graduate类对象中的公共基类Person的数据成员name、sex、age的值,不需要加基类名和域运算符(::),不会产生二义性。

可以看到:使用多重继承时要十分小心,经常会出现二义性问题。前面介绍的例子是简单的,如果派生的层次再多一些,多重继承更复杂一些,程序设计人员很容易陷人迷 魂阵,程序的编写、调试和维护工作都会变得更加困难。因此,许多专业人员认为:不要提倡在程序中使用多重继承,只有在比较简单和不易出现二义性的情况或实在必要时才使用多重继承,能用单一继承解决的问题就不要使用多重继承。也是由于这个原因,有些面向对象的程序设计语言(如Java,Smalltalk)并不支持多重继承。

文章出处:http://see.xidian.edu.cn/cpp/biancheng/view/238.html

转载于:https://www.cnblogs.com/zwj911112/p/4022697.html

C++虚基类详解(转)相关推荐

  1. C++虚继承和虚基类详解(二)

    虚继承(Virtual Inheritance) 为了解决多继承时的命名冲突和冗余数据问题,C++ 提出了虚继承,使得在派生类中只保留一份间接基类的成员. 在继承方式前面加上 virtual 关键字就 ...

  2. C++虚继承和虚基类详解(一)

    多继承(Multiple Inheritance)是指从多个直接基类中产生派生类的能力,多继承的派生类继承了所有父类的成员.尽管概念上非常简单,但是多个基类的相互交织可能会带来错综复杂的设计问题,命名 ...

  3. 详解虚函数的实现过程之虚基类(4)

    博客虚函数实现过程3 时提到过虚基类,这里呢,我们来详细讲述一下: 当我们在虚函数的声明结尾处添加"=0",这种虚函数就被称为纯虚函数. 它好似一个没有实现只有声明的函数,它的存在 ...

  4. c++ 虚函数多态、纯虚函数、虚函数表指针、虚基类表指针详解

    文章目录 静态多态.动态多态 虚函数 哪些函数类型不可以被定义成虚函数? 虚函数的访问方式 析构函数中的虚函数 虚函数表指针 vptr 多继承下的虚函数表 虚基类表指针 bptr 纯虚函数 抽象类 虚 ...

  5. 38.【C++ 虚函数 纯虚函数 虚基类 (最全详解)】

    虚函数.虚基类.纯虚函数 (一).虚函数 1.什么是虚函数: 2.虚函数的格式: 3.关于虚函数的注意事项: 4.虚函数的作用: 5.虚函数访问格式 6.虚函数的各种疑难杂症 [当指针是基类.但虚函数 ...

  6. C++ 多继承类 虚基类

    版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明. 本文链接:https://blog.csdn.net/jzj_c_love/article/d ...

  7. OpenCV Mat类详解和用法(官网原文)

    参考文章:OpenCV Mat类详解和用法 我马克一下,日后更 官网原文链接:https://docs.opencv.org/3.2.0/d6/d6d/tutorial_mat_the_basic_i ...

  8. [NewLife.XCode]实体类详解

    NewLife.XCode是一个有10多年历史的开源数据中间件,由新生命团队(2002~2019)开发完成并维护至今,以下简称XCode. 整个系列教程会大量结合示例代码和运行日志来进行深入分析,蕴含 ...

  9. OpenCV Mat类详解和用法

    OpenCV Mat类详解和用法 我们有多种方法可以获得从现实世界的数字图像:数码相机.扫描仪.计算机体层摄影或磁共振成像就是其中的几种.在每种情况下我们(人类)看到了什么是图像.但是,转换图像到我们 ...

  10. C++虚继承中的虚基类表

    虚继承主要解决多重继承会在子类中存在多份拷贝的问题,这不仅浪费空间,而且存在二义性. 在之前的 C++ 继承中已经说过虚继承基本概念,这里不再赘述.这篇文章主要探究虚继承的原理.文章中多处给出了类实例 ...

最新文章

  1. 著名站点的爬虫 —— 豆瓣
  2. ICCV 2019 | 可选择性与不变性:关注边界的显著性目标检测
  3. Spring @Value注解无法正确赋值问题
  4. [蓝桥杯2017初赛]算式900+dfs,next_permutation
  5. python打开浏览器后带cookie_Python爬虫使用浏览器的cookies:browsercookie
  6. 分数优先遵循志愿php源码_2021年南昌中考志愿填报指导,这七大要点你都了解吗?...
  7. Windows完全卸载oracle11g步骤
  8. Ubuntu下qt5无法输入中文
  9. ls命令具有一个-r选项,可以递归的列出子目录中的内容。请编写一个具有同样功能的程序。...
  10. matlab的libsvm工具箱(faruto版本)配置问题汇总,用于VS2015+R2017a
  11. VFS之基本数据结构
  12. 修改文件属性与权限(鸟哥linux私房菜)
  13. 计算机电源 3842,UC3842开关电源保护的几个技巧及电路图
  14. 吉他音阶训练——问题解答
  15. 野蛮人传教士问题(上)
  16. PS小知识(三)——画圆滑线及虚线
  17. uva-10041-水题
  18. 商场三十六计——第8计 “暗渡陈仓”
  19. Python预测基金净值:keras神经网络
  20. Android Study Material Design 十 再探沉浸式

热门文章

  1. DBeaverUE for Mac(数据库管理软件)旗舰版
  2. 快捷笔记应用Side Notes Mac
  3. new Date()浏览器兼容性问题
  4. iOS内存管理学习笔记二
  5. Oracle 正则表达式函数-REGEXP_SUBSTR 使用例子
  6. Java SE作业:判断一个字符串是否是视频文件
  7. java.io读写文本
  8. Flink技术到底是什么?Flink原理及深度解析
  9. Redis 那么快之底层 ziplist 的奥秘!
  10. Spring的核心思想,这篇文章短小精悍的总结透了