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

发布时间 - 2026-01-11 02:38:38    点击率:

前言

菱形继承是多重继承中跑不掉的,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++菱形继承和虚继承的实现  # 子类  # 当我们  # 只有一个  # 派生类  # 但是在  # 设置为  # 的是  # 时称  # 就会  # 使用了  # 放在  # 相关内容  # 在此  # 说了  # 不多  # 中有  # 并在  # 提出了  # 很容易  # 能在 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Laravel如何设置定时任务(Cron Job)_Laravel调度器与任务计划配置  Laravel如何使用Gate和Policy进行权限控制_Laravel权限判定与策略规则配置  Laravel如何部署到服务器_线上部署Laravel项目的完整流程与步骤  Laravel怎么处理异常_Laravel自定义异常处理与错误页面教程  javascript中闭包概念与用法深入理解  千库网官网入口推荐 千库网设计创意平台入口  高防服务器租用指南:配置选择与快速部署攻略  如何在VPS电脑上快速搭建网站?  如何为不同团队 ID 动态生成多个独立按钮  html5如何实现懒加载图片_ intersectionobserver api用法【教程】  Laravel中Service Container是做什么的_Laravel服务容器与依赖注入核心概念解析  如何在阿里云虚拟服务器快速搭建网站?  香港服务器网站搭建教程-电商部署、配置优化与安全稳定指南  Win11怎么修改DNS服务器 Win11设置DNS加速网络【指南】  如何用景安虚拟主机手机版绑定域名建站?  lovemo网页版地址 lovemo官网手机登录  Laravel用户认证怎么做_Laravel Breeze脚手架快速实现登录注册功能  android nfc常用标签读取总结  打开php文件提示内存不足_怎么调整php内存限制【解决方案】  Python3.6正式版新特性预览  高性能网站服务器配置指南:安全稳定与高效建站核心方案  如何在浏览器中启用Flash_2025年继续使用Flash Player的方法【过时】  Laravel如何生成URL和重定向?(路由助手函数)  C++时间戳转换成日期时间的步骤和示例代码  node.js报错:Cannot find module 'ejs'的解决办法  浅析上传头像示例及其注意事项  如何正确选择百度移动适配建站域名?  ChatGPT常用指令模板大全 新手快速上手的万能Prompt合集  如何在阿里云虚拟主机上快速搭建个人网站?  Midjourney怎样加参数调细节_Midjourney参数调整技巧【指南】  如何在云服务器上快速搭建个人网站?  青岛网站建设如何选择本地服务器?  ChatGPT回答中断怎么办 引导AI继续输出完整内容的方法  ,南京靠谱的征婚网站?  哪家制作企业网站好,开办像阿里巴巴那样的网络公司和网站要怎么做?  胶州企业网站制作公司,青岛石头网络科技有限公司怎么样?  想要更高端的建设网站,这些原则一定要坚持!  Laravel如何实现本地化和多语言支持_Laravel多语言配置与翻译文件管理  如何在云主机上快速搭建多站点网站?  使用spring连接及操作mongodb3.0实例  济南网站建设制作公司,室内设计网站一般都有哪些功能?  ,怎么在广州志愿者网站注册?  如何在HTML表单中获取用户输入并用JavaScript动态控制复利计算循环  Laravel怎么多语言本地化设置_Laravel语言包翻译与Locale动态切换【手册】  linux写shell需要注意的问题(必看)  如何将凡科建站内容保存为本地文件?  打造顶配客厅影院,这份100寸电视推荐名单请查收  Laravel如何处理JSON字段_Eloquent原生JSON字段类型操作教程  今日头条AI怎样推荐抢票工具_今日头条AI抢票工具推荐算法与筛选【技巧】  奇安信“盘古石”团队突破 iOS 26.1 提权