关于C++中菱形继承和虚继承的问题总结

 更新时间:2017年08月04日 09:43:13   作者:Suhw  
C++的三大特性为:封装,继承,多态。但是在继承中,存在一些使用方面的问题需要注意,下面这篇文章主要给大家总结介绍了关于C++中菱形继承和虚继承的问题,需要的朋友可以参考借鉴,下面来一起看看吧。

前言

菱形继承是多重继承中跑不掉的,Java拿掉了多重继承,辅之以接口。C++中虽然没有明确说明接口这种东西,但是只有纯虚函数的类可以看作Java中的接口。在多重继承中建议使用“接口”,来避免多重继承中可能出现的各种问题。本文将给大家详细介绍关于C++菱形继承和虚继承的相关内容,分享出来供大家参考学习,话不多说了,来一起看看详细的介绍吧。

继承:

      1. 单继承–一个子类只有一个直接父类时称这个继承关系为单继承

      2. 多继承–一个子类有两个或以上直接父类时称这个继承关系为多继承

例如下面这两个例子:

例一(单继承):

class A
{
public:
 int _a;
};

class B : public A // B是 子类/派生类, 公有 继承父类/基类 A
{
public:
 int _b;
};

class C : public B //C是 子类/派生类, 公有继承 父类/基类 B
{
public:
 int _c;
};

例二(多继承):

class A
{
public:
 int _a;
};

class B 
{
public:
 int _b;
};

class C : public A , public B // 子类C同时公有继承父类A和父类B
{
public:
 int _c;
};

用图很形象的表示一下:

但是在使用过程中,很容易出现一种继承关系叫菱形继承。就好比下面这种继承方式。

class A
{
public:
 int _a;
};

class B : public A
{
public:
 int _b;
};

class C : public A
{
public:
 int _c;
};

class D : public B, public C
{
public:
 int _d;
};

继承的方式简单画出来就是下面这样:

我们在使用过程中会发现以下缺点:

      1、 当我们用D类创建出对象d时,可以访问到_a,但是一旦编译就会出现错误。错误说明为: C2385: 对“_a”的访问不明确。从图中也可以看出,如果用d访问_a时,可能在B类里,也同时可能存在于c类中。这就是所谓的“二义性”;

      2、虽然B类和C类都公有继承A,但是在D类公有继承B,C时,存放了两份A类,造成了数据的冗余。

C++针对这种缺陷提出了另外一种继承方式叫做虚继承。

虚继承

C++使用虚拟继承(Virtual Inheritance),解决从不同途径继承来的同名的数据成员在内存中有不同的拷贝造成数据不一致问题,将共同基类设置为虚基类。这时从不同的路径继承过来的同名数据成员在内存中就只有一个拷贝,同一个函数名也只有一个映射。

◇语法:

class 派生类: virtual 基类1,virtual 基类2,…,virtual 基类n

{

…//派生类成员声明

};

在有了虚继承的概念后,我们就可以规避上面的缺点了。

class A
{
public:
 int _a;
};

class B : virtual public A
{
public:
 int _b;
};

class C : virtual public A
{
public:
 int _c;
};

class D : public B, public C
{
public:
 int _d;
};

当我们使用了虚继承时,继承模型就改变为下面这样:

由于我所使用的是vs2015,在此编译器下对应的处理方式就是这样。将class B 和 class C设置为虚继承后,编译器将class A存放在了最下端,并在B和C类的前四个字节中存放了一个地址,当我们访问过去向下再多看四个字节时就会发现这其中存放了一个数字。而这个数字就类似于“偏移量”,记录了该类的首地址距父类首地址之间的字节差距。比如class B中,我们找到对应数字为14,但是这个数字是16进制,转为10进制为20,在class B的首地址加上20个字节就恰好是class A的首地址,同理class C。

因此在class D访问_a时,就不会产生二义性,_a数据也只存放了一份,解决了之前菱形继承所带来的问题。

但是还存在一个问题:当我们求没有使用虚继承之前的class D的大小,结果是20,但是在使用了虚继承后大小变为24。所以虽然使用虚继承解决数据冗余问题也带来了性能上的损耗。(关于如何计算内存大小,可以参考此链接。)

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对脚本之家的支持。

您可能感兴趣的文章:

相关文章

  • 适合初学者的C语言转义字符讲解

    适合初学者的C语言转义字符讲解

    转义字符是很多程序语言、数据格式和通信协议的形式文法的一部分。对于一个给定的字母表,一个转义字符的目的是开始一个字符序列,使得转义字符开头的该字符序列具有不同于该字符序列单独出现(没有转义字符开头)时的语义。因此转义字符开头的字符序列被叫做转义序列
    2022-04-04
  • 详解Matlab如何绘制圆角半透明图例

    详解Matlab如何绘制圆角半透明图例

    目前MATLAB的legend图例是不支持圆角和半透明的,所以本文将自制实现圆角半透明图例。文中的示例代码讲解详细,需要的可以参考一下
    2022-05-05
  • c语言合并两个已排序数组的示例(c语言数组排序)

    c语言合并两个已排序数组的示例(c语言数组排序)

    如何将两个已排序数组合并成一个排序数组,下面我们给出使用c语言合并两个已排序数组的示例,需要的朋友可以参考下
    2014-03-03
  • C语言实现遍历文件夹中的文件

    C语言实现遍历文件夹中的文件

    这篇文章主要为大家详细介绍了如何使用C语言实现遍历文件夹中的文件,文中的示例代码讲解详细,具有一定的借鉴价值,有需要的小伙伴可以参考一下
    2024-02-02
  • C语言基于EasyX绘制时钟

    C语言基于EasyX绘制时钟

    这篇文章主要为大家详细介绍了C语言基于EasyX绘制时钟,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-06-06
  • 判断两颗二叉树是否相似的两种方法

    判断两颗二叉树是否相似的两种方法

    今天小编就为大家分享一篇关于判断两颗二叉树是否相似的两种方法,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2019-03-03
  • 详解C++中的常量

    详解C++中的常量

    这篇文章主要介绍了C++中的常量的相关资料,文中示例代码非常详细,帮助大家更好的理解和学习,感兴趣的朋友可以了解下
    2020-07-07
  • C语言实现走迷宫

    C语言实现走迷宫

    这篇文章主要为大家详细介绍了C语言实现走迷宫,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-07-07
  • C++之list容器介绍及使用方式

    C++之list容器介绍及使用方式

    这篇文章主要介绍了C++之list容器介绍及使用方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-02-02
  • C++操作json文件以及jsoncpp配置详解

    C++操作json文件以及jsoncpp配置详解

    这篇文章主要给大家介绍了关于C++操作json文件以及jsoncpp配置的相关资料,文中通过实例代码及图片介绍的非常详细,需要的朋友可以参考下
    2021-06-06

最新评论