C++菱形继承及解决方法

2024-05-28 10:38
文章标签 c++ 方法 解决 继承 菱形

本文主要是介绍C++菱形继承及解决方法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1.何为菱形继承?

两个子类继承同一个父类,而又有子类又分别继承这两个子类,就如上图说示。

复制代码

#include<stdio.h>
#include<iostream>
#include<queue>
using namespace std;class A {
public:A(){printf("A create.\n");}int a;virtual void fun(){}
};class B: public A{
public:B(){printf("B create.\n");}int b;virtual void fun1(){}
};class C: public A
{
public :int c;C(){printf("C create.\n");}virtual void fun3(){printf("fun3 .\n");}
};class D:public C,public B{
public:int d;D(){printf("D create.\n");}virtual void fun3(){printf("fun4 .\n");}
}; 
//二义性问题的开销 
int main() {D *pd=new D;printf("%d\n",sizeof(D));getchar();
}

复制代码

产生的问题,会产生二义性问题,即对于baseClass的调用要说明作用域的情况:

    D *pd=new D;pd->B::a=1;pd->C::a=2;printf("%d\n",pd->B::a);printf("%d\n",pd->C::a);

相当于baseClass在类中有两个,这可能不是我们想要的结果,增加调用的困难,同时也会浪费内存资源。

这种结构如图:

可以看到A指向的虚函数表的位置是不一样的!即baseClass有两个实例!

2.如何解决?

使用虚拟继承!

复制代码

#include<stdio.h>
#include<iostream>
#include<queue>
using namespace std;class A {
public:A(){printf("A create.\n");}int a;virtual void fun(){}
};class B:virtual public A{
public:B(){printf("B create.\n");}int b;virtual void fun1(){}
};class C:virtual public A
{
public :int c;C(){printf("C create.\n");}virtual void fun3(){printf("fun3 .\n");}
};class D:public C,public B{
public:int d;D(){printf("D create.\n");}virtual void fun3(){printf("fun4 .\n");}
}; 
//二义性问题的开销 
int main() {D *pd=new D;pd->B::a=1;pd->C::a=2;printf("%d\n",pd->B::a);printf("%d\n",pd->C::a);printf("%d\n",sizeof(D));getchar();
}

复制代码

内存布局:

  对于baseClass是公用的,也就是baseClass就实例化了一个对象!想想这会有什么后果?调用B,C的虚函数的时候就一个虚表怎么行,所以有需要对应有两个相应的虚表指向B,C,于是就成了上面的结构了。

  调试观察,果然如此!

  

总结:可以通过虚拟继承消除二义性,但是虚拟继承的开销是增加虚函数指针。

参考:C++程序优化

 

ps:这时就有疑问了,既然多重继承的会有二义性的问题,为什么编译器不能自己识别并处理呢,也许别人给你写的一个类中与你写的类中同时继承了一个基类(不知情),别人又同时继承了这两个类= =!可能有人会说,多重继承太复杂,一般都用单一继承!也许这就是C++优势有时也是劣势吧= =

这篇关于C++菱形继承及解决方法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Conda与Python venv虚拟环境的区别与使用方法详解

《Conda与Pythonvenv虚拟环境的区别与使用方法详解》随着Python社区的成长,虚拟环境的概念和技术也在不断发展,:本文主要介绍Conda与Pythonvenv虚拟环境的区别与使用... 目录前言一、Conda 与 python venv 的核心区别1. Conda 的特点2. Python v

Spring Boot中WebSocket常用使用方法详解

《SpringBoot中WebSocket常用使用方法详解》本文从WebSocket的基础概念出发,详细介绍了SpringBoot集成WebSocket的步骤,并重点讲解了常用的使用方法,包括简单消... 目录一、WebSocket基础概念1.1 什么是WebSocket1.2 WebSocket与HTTP

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

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

MyBatis-Plus 中 nested() 与 and() 方法详解(最佳实践场景)

《MyBatis-Plus中nested()与and()方法详解(最佳实践场景)》在MyBatis-Plus的条件构造器中,nested()和and()都是用于构建复杂查询条件的关键方法,但... 目录MyBATis-Plus 中nested()与and()方法详解一、核心区别对比二、方法详解1.and()

golang中reflect包的常用方法

《golang中reflect包的常用方法》Go反射reflect包提供类型和值方法,用于获取类型信息、访问字段、调用方法等,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值... 目录reflect包方法总结类型 (Type) 方法值 (Value) 方法reflect包方法总结

C# 比较两个list 之间元素差异的常用方法

《C#比较两个list之间元素差异的常用方法》:本文主要介绍C#比较两个list之间元素差异,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1. 使用Except方法2. 使用Except的逆操作3. 使用LINQ的Join,GroupJoin

MySQL查询JSON数组字段包含特定字符串的方法

《MySQL查询JSON数组字段包含特定字符串的方法》在MySQL数据库中,当某个字段存储的是JSON数组,需要查询数组中包含特定字符串的记录时传统的LIKE语句无法直接使用,下面小编就为大家介绍两种... 目录问题背景解决方案对比1. 精确匹配方案(推荐)2. 模糊匹配方案参数化查询示例使用场景建议性能优

关于集合与数组转换实现方法

《关于集合与数组转换实现方法》:本文主要介绍关于集合与数组转换实现方法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、Arrays.asList()1.1、方法作用1.2、内部实现1.3、修改元素的影响1.4、注意事项2、list.toArray()2.1、方

Python中注释使用方法举例详解

《Python中注释使用方法举例详解》在Python编程语言中注释是必不可少的一部分,它有助于提高代码的可读性和维护性,:本文主要介绍Python中注释使用方法的相关资料,需要的朋友可以参考下... 目录一、前言二、什么是注释?示例:三、单行注释语法:以 China编程# 开头,后面的内容为注释内容示例:示例:四

从入门到精通C++11 <chrono> 库特性

《从入门到精通C++11<chrono>库特性》chrono库是C++11中一个非常强大和实用的库,它为时间处理提供了丰富的功能和类型安全的接口,通过本文的介绍,我们了解了chrono库的基本概念... 目录一、引言1.1 为什么需要<chrono>库1.2<chrono>库的基本概念二、时间段(Durat