C++ 菱形继承与虚拟继承的详解与代码示例

2024-09-06 17:36

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

在C++中,多重继承虽然强大,但也会带来不少问题。特别是当继承链中出现菱形继承时,容易导致基类的重复实例化。本文将深入讨论菱形继承的问题,并详细解释如何通过虚拟继承来解决它。同时,给出一个简单的代码示例,并逐步解析输出结果。

什么是菱形继承?

菱形继承是指在多重继承中,同一个基类被多个派生类继承,而这些派生类又被另一个类继承,最终形成菱形结构,如下图所示:

    A/ \B   C\ /D
  • BC 都继承了基类 A
  • D 继承了 BC

在这种结构中,D 类通过 BC 继承了 A,这意味着 D 类中可能会存在两个 A 类的实例,导致数据重复、内存浪费以及二义性问题。

菱形继承的问题

多重基类实例:当 D 类继承自 BC 时,由于 BC 都继承了 A,所以 D 类中有两个 A 的实例。如下所示:

class A {
public:int data;
};class B : public A {};
class C : public A {};
class D : public B, public C {};int main() {D d;// 访问 A 类的成员变量 data 会导致二义性问题d.data = 10;  // 错误:编译器无法确定该选择 B::A 还是 C::A
}

这种情况下,编译器会提示二义性错误,因为无法确定 d.data 是来自 B 中的 A 还是 C 中的 A

如何通过虚拟继承解决?

C++ 提供了虚拟继承virtual inheritance)来解决菱形继承中基类实例重复的问题。通过在继承基类时使用 virtual 关键字,可以确保基类的实例在最终派生类中只存在一份。

虚拟继承语法

class B : virtual public A {};
class C : virtual public A {};

BC 通过虚拟继承自 A 时,A 的实例不会被重复创建,D 类中的 A 实例只会有一份。

虚拟继承的代码示例

#include <iostream>// 基类 A
class A {
public:int data;A() : data(0) {std::cout << "A's constructor called" << std::endl;  // 输出 1}void show() {std::cout << "A::data = " << data << std::endl;  // 输出 6}
};// 派生类 B 虚拟继承自 A
class B : virtual public A {
public:B() {std::cout << "B's constructor called" << std::endl;  // 输出 2}
};// 派生类 C 虚拟继承自 A
class C : virtual public A {
public:C() {std::cout << "C's constructor called" << std::endl;  // 输出 3}
};// 最终派生类 D 同时继承自 B 和 C
class D : public B, public C {
public:D() {std::cout << "D's constructor called" << std::endl;  // 输出 4}
};int main() {D d;  // 创建 D 类对象,依次调用 A、B、C、D 的构造函数// 输出顺序为:A's constructor -> B's constructor -> C's constructor -> D's constructord.data = 100;  // 修改 A 中的成员变量 datad.show();  // 调用 A 的 show 函数,输出 A::data = 100return 0;
}

输出结果

A's constructor called   // 由 A 的构造函数触发,这是基类的构造顺序,最先被调用。
B's constructor called   // 由 B 的构造函数触发,B 虚拟继承 A,但 A 已经构造过了。
C's constructor called   // 由 C 的构造函数触发,C 虚拟继承 A,A 仍然只会构造一次。
D's constructor called   // 由 D 的构造函数触发,最后构造 D。
A::data = 100            // 调用 A 的 show 函数,输出 A::data 的值为 100。

详细解析

  1. A's constructor calledD 类对象创建时,首先调用 A 类的构造函数。因为 BC 都虚拟继承自 A,所以 A 的实例只被构造一次。
  2. B's constructor called:接着调用 B 的构造函数,但 A 不会再次构造。
  3. C's constructor called:然后调用 C 的构造函数,同样 A 不会再次构造。
  4. D's constructor called:最后调用 D 的构造函数。
  5. A::data = 100:调用 d.show() 方法,输出 A::data 的值,验证修改操作生效。

总结

  • 菱形继承容易导致基类的重复实例化和二义性问题。
  • 虚拟继承通过确保基类只存在一个实例,解决了多重基类实例化的问题。
  • 虚拟继承虽然增加了一些内存开销,但有效避免了继承结构中的潜在冲突和复杂性。

在复杂的继承结构中,虚拟继承是非常有用的,特别是在需要共享基类实例的情况下。

这篇关于C++ 菱形继承与虚拟继承的详解与代码示例的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

OpenHarmony鸿蒙开发( Beta5.0)无感配网详解

1、简介 无感配网是指在设备联网过程中无需输入热点相关账号信息,即可快速实现设备配网,是一种兼顾高效性、可靠性和安全性的配网方式。 2、配网原理 2.1 通信原理 手机和智能设备之间的信息传递,利用特有的NAN协议实现。利用手机和智能设备之间的WiFi 感知订阅、发布能力,实现了数字管家应用和设备之间的发现。在完成设备间的认证和响应后,即可发送相关配网数据。同时还支持与常规Sof

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

活用c4d官方开发文档查询代码

当你问AI助手比如豆包,如何用python禁止掉xpresso标签时候,它会提示到 这时候要用到两个东西。https://developers.maxon.net/论坛搜索和开发文档 比如这里我就在官方找到正确的id描述 然后我就把参数标签换过来

06 C++Lambda表达式

lambda表达式的定义 没有显式模版形参的lambda表达式 [捕获] 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 有显式模版形参的lambda表达式 [捕获] <模版形参> 模版约束 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 含义 捕获:包含零个或者多个捕获符的逗号分隔列表 模板形参:用于泛型lambda提供个模板形参的名

poj 1258 Agri-Net(最小生成树模板代码)

感觉用这题来当模板更适合。 题意就是给你邻接矩阵求最小生成树啦。~ prim代码:效率很高。172k...0ms。 #include<stdio.h>#include<algorithm>using namespace std;const int MaxN = 101;const int INF = 0x3f3f3f3f;int g[MaxN][MaxN];int n

6.1.数据结构-c/c++堆详解下篇(堆排序,TopK问题)

上篇:6.1.数据结构-c/c++模拟实现堆上篇(向下,上调整算法,建堆,增删数据)-CSDN博客 本章重点 1.使用堆来完成堆排序 2.使用堆解决TopK问题 目录 一.堆排序 1.1 思路 1.2 代码 1.3 简单测试 二.TopK问题 2.1 思路(求最小): 2.2 C语言代码(手写堆) 2.3 C++代码(使用优先级队列 priority_queue)