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

相关文章

Java NoClassDefFoundError运行时错误分析解决

《JavaNoClassDefFoundError运行时错误分析解决》在Java开发中,NoClassDefFoundError是一种常见的运行时错误,它通常表明Java虚拟机在尝试加载一个类时未能... 目录前言一、问题分析二、报错原因三、解决思路检查类路径配置检查依赖库检查类文件调试类加载器问题四、常见

解决IDEA报错:编码GBK的不可映射字符问题

《解决IDEA报错:编码GBK的不可映射字符问题》:本文主要介绍解决IDEA报错:编码GBK的不可映射字符问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录IDEA报错:编码GBK的不可映射字符终端软件问题描述原因分析解决方案方法1:将命令改为方法2:右下jav

C#如何调用C++库

《C#如何调用C++库》:本文主要介绍C#如何调用C++库方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录方法一:使用P/Invoke1. 导出C++函数2. 定义P/Invoke签名3. 调用C++函数方法二:使用C++/CLI作为桥接1. 创建C++/CL

Java 中的 @SneakyThrows 注解使用方法(简化异常处理的利与弊)

《Java中的@SneakyThrows注解使用方法(简化异常处理的利与弊)》为了简化异常处理,Lombok提供了一个强大的注解@SneakyThrows,本文将详细介绍@SneakyThro... 目录1. @SneakyThrows 简介 1.1 什么是 Lombok?2. @SneakyThrows

MyBatis模糊查询报错:ParserException: not supported.pos 问题解决

《MyBatis模糊查询报错:ParserException:notsupported.pos问题解决》本文主要介绍了MyBatis模糊查询报错:ParserException:notsuppo... 目录问题描述问题根源错误SQL解析逻辑深层原因分析三种解决方案方案一:使用CONCAT函数(推荐)方案二:

判断PyTorch是GPU版还是CPU版的方法小结

《判断PyTorch是GPU版还是CPU版的方法小结》PyTorch作为当前最流行的深度学习框架之一,支持在CPU和GPU(NVIDIACUDA)上运行,所以对于深度学习开发者来说,正确识别PyTor... 目录前言为什么需要区分GPU和CPU版本?性能差异硬件要求如何检查PyTorch版本?方法1:使用命

Qt实现网络数据解析的方法总结

《Qt实现网络数据解析的方法总结》在Qt中解析网络数据通常涉及接收原始字节流,并将其转换为有意义的应用层数据,这篇文章为大家介绍了详细步骤和示例,感兴趣的小伙伴可以了解下... 目录1. 网络数据接收2. 缓冲区管理(处理粘包/拆包)3. 常见数据格式解析3.1 jsON解析3.2 XML解析3.3 自定义

SpringMVC 通过ajax 前后端数据交互的实现方法

《SpringMVC通过ajax前后端数据交互的实现方法》:本文主要介绍SpringMVC通过ajax前后端数据交互的实现方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价... 在前端的开发过程中,经常在html页面通过AJAX进行前后端数据的交互,SpringMVC的controll

Java中的工具类命名方法

《Java中的工具类命名方法》:本文主要介绍Java中的工具类究竟如何命名,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录Java中的工具类究竟如何命名?先来几个例子几种命名方式的比较到底如何命名 ?总结Java中的工具类究竟如何命名?先来几个例子JD

Spring Security自定义身份认证的实现方法

《SpringSecurity自定义身份认证的实现方法》:本文主要介绍SpringSecurity自定义身份认证的实现方法,下面对SpringSecurity的这三种自定义身份认证进行详细讲解,... 目录1.内存身份认证(1)创建配置类(2)验证内存身份认证2.JDBC身份认证(1)数据准备 (2)配置依