利用虚继承解决菱形继承(钻石继承)的问题 学习笔记

2023-12-20 05:28

本文主要是介绍利用虚继承解决菱形继承(钻石继承)的问题 学习笔记,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

菱形继承概念:

两个派生类继承同一个基类

又有某个类同时继承两个派生类

这种继承又被称为菱形继承,或者钻石继承

定义一个基类Animal,两个派生类Sheep、Camal,SheepCame继承于两个派生类

class Animal
{
public:int m_Age;
};
class Sheep :  public Animal { };
class Camel :  public Animal { };
class SheepCamel : public Sheep, public Camel { };

void test()
{
    SheepCamel S;
    S.m_Age = 20;//报错,"SheepCamel::m_Age不明确"
}

问题(1):

当创建SheepCamel类对象,直接初始化m_Age时,会出现二义性,即不知是继承于Sheep父类还是Camel父类

若想初始化m_Age,则需要加作用域来强调继承于哪个父类

void test()
{
    SheepCamel S;
    S.Sheep::m_Age = 10;
    S.Camel::m_Age = 20;
    cout << S.Sheep::m_Age << endl; // 10
    cout << S.Camel::m_Age << endl;//  20
}

问题(2):

菱形继承中,SheepCamel子类会继承两份数据,即羊驼获得了10和20两个年龄;这造成了资源浪费,因为我们只需要一份数据,即羊驼是不应该有两个年龄的

解决方法:

利用虚继承即可解决问题。即:将共同基类设置为虚基类

虚基类的定义:class 派生类名 : virtual 继承方式 基类名

class Animal
{
public:int m_Age;
};
class Sheep : virtual public Animal { };
class Camel : virtual public Animal { };
class SheepCamel : public Sheep, public Camel{ };

{
    SheepCamel S;
    S.m_Age = 20;
    S.Sheep::m_Age = 30;
    S.Camel::m_Age = 50;
    cout << "Sheep类中:" << S.Sheep::m_Age << endl;// 50
    cout << "Camel类中:" << S.Camel::m_Age << endl;// 50
    cout << "SheepCamel类中:" << S.m_Age << endl; // 50
}

此时:数据只有一份,类似静态变量,无论是用子类调用还是父类调用都是共享同一份数据。也可以理解为:不管给哪个赋值都是给同一个变量赋值,所以最后等于后赋的值

底层原理:

我们可以通过对象模型来直观理解

在visual studio code  下:利用开发人员命令提示工具查看对象模型(我使用的是vs2022)

1.找到开发人员命令提示工具

2.输入 cd空格+当前源文件的路径(若是C盘,直接输入即可,若是F盘,先 F:转入F盘)

当前源文件的路径获取方法:

最后:输入到开发人员命令提示工具 (输入 cd空格+当前源文件的路径)

3.输入dir,进行查看当前目录是否有当前源文件

4.有当前源文件,继续查找当前SheepCamel类的对象模型布局

输入:cl /d1 reportSingleClassLayout+查找类名 当前源文件名

输入:cl /d1 reportSingleClassLayoutSheepCamel 源.cpp

未设置为虚基类时:

我们可以发现Sheep类的字节大小为8个字节,分别继承于Sheep类、Camel类的成员变量m_Age;

的确是继承了两份数据。

设置为虚基类时:

我们可以发现Sheep类的字节大小为12个字节,分别是Sheep类的虚基类指针、Camel类的虚基类指针,子类继承的m_Age;

此时,只继承了一份数据。

总结:在虚基类中,继承仅有的一份数据存储于子类,可以通过 虚基类指针访问子类来修改其数据的值,或者 子类对象本身来修改。

这篇关于利用虚继承解决菱形继承(钻石继承)的问题 学习笔记的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/514892

相关文章

SQL Server配置管理器无法打开的四种解决方法

《SQLServer配置管理器无法打开的四种解决方法》本文总结了SQLServer配置管理器无法打开的四种解决方法,文中通过图文示例介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的... 目录方法一:桌面图标进入方法二:运行窗口进入检查版本号对照表php方法三:查找文件路径方法四:检查 S

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Java 线程安全与 volatile与单例模式问题及解决方案

《Java线程安全与volatile与单例模式问题及解决方案》文章主要讲解线程安全问题的五个成因(调度随机、变量修改、非原子操作、内存可见性、指令重排序)及解决方案,强调使用volatile关键字... 目录什么是线程安全线程安全问题的产生与解决方案线程的调度是随机的多个线程对同一个变量进行修改线程的修改操

Redis出现中文乱码的问题及解决

《Redis出现中文乱码的问题及解决》:本文主要介绍Redis出现中文乱码的问题及解决,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1. 问题的产生2China编程. 问题的解决redihttp://www.chinasem.cns数据进制问题的解决中文乱码问题解决总结

全面解析MySQL索引长度限制问题与解决方案

《全面解析MySQL索引长度限制问题与解决方案》MySQL对索引长度设限是为了保持高效的数据检索性能,这个限制不是MySQL的缺陷,而是数据库设计中的权衡结果,下面我们就来看看如何解决这一问题吧... 目录引言:为什么会有索引键长度问题?一、问题根源深度解析mysql索引长度限制原理实际场景示例二、五大解决

Springboot如何正确使用AOP问题

《Springboot如何正确使用AOP问题》:本文主要介绍Springboot如何正确使用AOP问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录​一、AOP概念二、切点表达式​execution表达式案例三、AOP通知四、springboot中使用AOP导出

C++11委托构造函数和继承构造函数的实现

《C++11委托构造函数和继承构造函数的实现》C++引入了委托构造函数和继承构造函数这两个重要的特性,本文主要介绍了C++11委托构造函数和继承构造函数的实现,具有一定的参考价值,感兴趣的可以了解一下... 目录引言一、委托构造函数1.1 委托构造函数的定义与作用1.2 委托构造函数的语法1.3 委托构造函

Python中Tensorflow无法调用GPU问题的解决方法

《Python中Tensorflow无法调用GPU问题的解决方法》文章详解如何解决TensorFlow在Windows无法识别GPU的问题,需降级至2.10版本,安装匹配CUDA11.2和cuDNN... 当用以下代码查看GPU数量时,gpuspython返回的是一个空列表,说明tensorflow没有找到

解决未解析的依赖项:‘net.sf.json-lib:json-lib:jar:2.4‘问题

《解决未解析的依赖项:‘net.sf.json-lib:json-lib:jar:2.4‘问题》:本文主要介绍解决未解析的依赖项:‘net.sf.json-lib:json-lib:jar:2.4... 目录未解析的依赖项:‘net.sf.json-lib:json-lib:jar:2.4‘打开pom.XM

Java 继承和多态的作用及好处

《Java继承和多态的作用及好处》文章讲解Java继承与多态的概念、语法及应用,继承通过extends复用父类成员,减少冗余;多态实现方法重写与向上转型,提升灵活性与代码复用性,动态绑定降低圈复杂度... 目录1. 继承1.1 什么是继承1.2 继承的作用和好处1.3 继承的语法1.4 子类访问父类里面的成