关于C++中菱形继承和虚继承的问题总结
作者:Suhw 发布时间:2021-10-03 20:22:23
前言
菱形继承是多重继承中跑不掉的,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。所以虽然使用虚继承解决数据冗余问题也带来了性能上的损耗。(关于如何计算内存大小,可以参考此链接。)
来源:http://blog.csdn.net/sssssuuuuu666/article/details/75736954


猜你喜欢
- Java是一种强类型, 许多流行的编程语言都已经支持局部变量类型推断,如js,Python,C++等JDK10 可以使用var作为局部变量类
- 本文实例讲述了Android中View的炸裂特效实现方法。分享给大家供大家参考,具体如下:前几天微博上被一个很优秀的 Android 开源组
- 一. 下载环境Ubuntu 2.x.x 版本二. 创建Hadoop用户在虚拟机创建安装完成后。1.进入用户,打开终端输入如下命令:sudo
- using System.Runtime.InteropServices;using System.Drawing.Imaging;&nbs
- 随机数的定义为:产生的所有数字毫无关系.在实际应用中很多地方会用到随机数,比如需要生成唯一的订单号.在C#中获取随机数有三种方法:一.Ran
- 本文实例讲述了Java基于Runtime调用外部程序出现阻塞的解决方法, 是一个很实用的技巧。分享给大家供大家参考。具体分析如下:有时候在j
- Mybatis多层嵌套查询三张表:user article blog表的存储sql文件/*Navicat MySQL Data Transf
- 其实说实话,没有多大的可比较性,它们是完全不同的两个东西,它们的抽象不在同一个层级上。但是为了让大家更好的理解,还是做一个比较吧,毕竟它们都
- java 中cookie的详解Java对cookie的操作比较简单,主要介绍下建立cookie和读取cookie,以及如何设定cookie的
- 1.创建一个redis maven项目,在pom中添加如下信息spring boot 版本 <parent> <group
- 为了防止测试妹子或者用户频繁点击某个按钮,导致程序在短时间内进行多次数据提交or数据处理,那到时候就比较坑了~那么如何有效避免这种情况的发生
- 本文介绍了springcloud Feign的Hystrix支持,分享给大家,具体如下:一、Feign client中加入Hystrix的f
- 在看了网上多篇rxjava和retrofit的文章后,大概有了一个初步的认识,刚好要做一个多图上传的功能,就拿它开刀吧。下面的内容将基于之前
- 大家都知道Android Studio可以直接在“Menu - Check for Updates...”自动检测并更新版本,还可以在弹出的
- 1 SeekBar简介SeekBar是进度条。我们使用进度条时,可以使用系统默认的进度条;也可以自定义进度条的图片和滑块图片等。2 Seek
- 当键盘敲下后退键(Backspace)后1、禁止浏览器自动后退2、但不影响密码、单行文本、多行文本输入框等的回退操作<script t
- 环境:apache-tomcat-8.5.15jdk1.8.0_172IDEA建立一个maven-webapp项目:Create New P
- @Value取值为NULL的问题在spring mvc架构中,如果希望在程序中直接使用properties中定义的配置值,通常使用一下方式来
- 前言:Timer是一个定时器,作为C#开发Timer控件是我们用的比较多的一个控件,它的功能很简单,但是也是值得我们去学习的一个知识点,今天
- 在学习安卓的最初过程中我们学的都是最基本的一个活动,只有一个活动的应用也太简单了吧,没错我们的最求应该更高点,不管你创建多少个活动,接下里我