当前位置: 首页 > news >正文

菱形继承原理

在C++中,菱形继承的内存模型会因是否使用虚继承产生本质差异。我们通过具体示例说明两种场景的区别:


一、普通菱形继承的内存模型

class A { int a; };
class B : public A { int b; };
class C : public A { int c; };
class D : public B, public C { int d; };

内存布局特点:

|-------------------|
| B::A::a (4字节)   |
| B::b (4字节)      |
|-------------------|
| C::A::a (4字节)   |
| C::c (4字节)      |
|-------------------|
| D::d (4字节)      |
|-------------------|

关键问题:

  1. 冗余存储:派生类D包含两份A的成员变量(B::A::a 和 C::A::a)
  2. 访问二义性d.a 需要明确指定路径(d.B::ad.C::a

二、虚继承后的内存模型

class A { int a; };
class B : virtual public A { int b; };
class C : virtual public A { int c; };
class D : public B, public C { int d; };

典型内存布局(以GCC为例):

|-------------------|
| B::vbptr (8字节*) | ➝ 虚基类表,记录A的偏移量
| B::b (4字节)      |
|-------------------|
| C::vbptr (8字节*) | ➝ 同样指向A的偏移量
| C::c (4字节)      |
|-------------------|
| D::d (4字节)      |
|-------------------|
| A::a (4字节)      | ← 唯一一份A的成员
| Padding (4字节)   | (对齐填充)
|-------------------|

核心变化:

  1. 共享基类:虚基类A的成员a在D中只有一份
  2. 间接访问:通过虚基类指针(vbptr)定位共享的A实例
  3. 初始化责任:D的构造函数直接初始化A

三、关键差异对比

特征普通继承虚继承
基类冗余存储存在两份A共享唯一A实例
派生类大小较大(含重复数据)较小但含指针开销
访问基类成员直接访问通过虚基类表间接访问
初始化方式中间类负责初始化最终派生类负责初始化

四、验证示例

#include <iostream>
using namespace std;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; };int main() {D d;d.B::a = 1;  // 虚继承后,修改的是同一份A::ad.C::a = 2;  cout << d.B::a;  // 输出2,证明A是共享的
}

五、注意:在虚继承情况下,虚基类的构造由最底层的派生类直接负责,而不是由中间的基类来构造的。

六、典型应用

在C++标准库中,经典的虚继承解决菱形继承的案例体现在输入输出流(iostream)库的实现中。以下是具体分析:


标准库中的流类继承体系
            basic_ios<...>↑     ↑虚|     ||     |basic_istream<...>  basic_ostream<...>↖       ↗basic_iostream<...>
关键结构解析
  1. **基类 **basic_ios
    所有流类的公共基类,负责管理流的状态(如错误标志、格式化设置等)。
  2. **中间派生类 basic_istream 和 **basic_ostream
    • basic_istream(输入流)通过虚继承派生自 basic_ios
    • basic_ostream(输出流)通过虚继承派生自 basic_ios
  3. **最终派生类 **basic_iostream
    同时继承 basic_istreambasic_ostream,需确保 basic_ios 仅存在一份实例。

虚继承的作用
  • 避免菱形继承的二义性
    basic_istreambasic_ostream 未虚继承 basic_ios,则 basic_iostream 将包含两个独立的 basic_ios 实例,导致访问公共成员(如 good()setf())时出现二义性。
  • 确保单一共享基类
    通过虚继承,basic_iostream 仅保留一个 basic_ios 实例,避免冗余存储和成员冲突。

验证虚继承的示例
#include <iostream>int main() {std::iostream& io = std::cin;  // 合法:std::cin是std::istream&,但向上转型安全io.get();  // 正确调用basic_ios的成员,无二义性return 0;
}
  • 构造顺序
    basic_iostream 的构造函数直接初始化虚基类 basic_ios,确保基类仅构造一次。

标准库实现代码片段(简化)
// 基类
template<typename CharT, typename Traits>
class basic_ios : public ios_base { /*...*/ };// 输入流(虚继承)
template<typename CharT, typename Traits>
class basic_istream : virtual public basic_ios<CharT, Traits> { /*...*/ };// 输出流(虚继承)
template<typename CharT, typename Traits>
class basic_ostream : virtual public basic_ios<CharT, Traits> { /*...*/ };// 最终流
template<typename CharT, typename Traits>
class basic_iostream : public basic_istream<CharT, Traits>,public basic_ostream<CharT, Traits> {
public:// 显式调用虚基类构造函数explicit basic_iostream(/*...*/) : basic_ios<CharT, Traits>(/*...*/),basic_istream<CharT, Traits>(/*...*/),basic_ostream<CharT, Traits>(/*...*/) {}
};

总结

  • 普通菱形继承:基类冗余存储,存在数据冗余和二义性。
  • 虚继承:通过虚基类指针共享唯一基类,牺牲间接访问性能换取空间和语义统一。编译器通过虚基类表(如GCC的vbptr)管理偏移量,确保派生类正确访问共享基类。
  • 最后,尽量不使用菱形继承:
    ● 组合代替继承:将共享功能封装为工具类,通过对象组合调用。
    ● 接口分离:将基类拆分为多个职责单一的接口,避免多重继承。
    ● 依赖注入:通过参数传递依赖对象,而非直接继承。
http://www.xdnf.cn/news/514603.html

相关文章:

  • java集合相关的api-总结
  • 2025年- H27-Lc135- 239.滑动窗口最大值(自定义双端队列)---java版
  • 量子计算在金融科技中的应用前景
  • [Codeforce刷题8]
  • 无废话离线大模型安装
  • 【随机过程】贝叶斯估计
  • 游戏引擎学习第292天:实现蛇
  • es聚合-词条统计
  • 量子计算 | 量子密码学的挑战和机遇
  • LWIP的NETCONN接口
  • APP手机端测试覆盖点
  • 专业漏洞扫描机构如何助力企业保障安全并提升竞争力?
  • 【MySQL】库与表的操作
  • 力扣热题——数组的最小相等和
  • 关于 Web 漏洞原理与利用:1. SQL 注入(SQLi)
  • 基于FPGA的电子万年历系统开发,包含各模块testbench
  • ​Docker 网络
  • 前端三剑客之HTML
  • 深入解析Python中的Vector2d类:从基础实现到特殊方法的应用
  • nginx服务器实验
  • 23种设计模式解释+记忆
  • 虚幻引擎5-Unreal Engine笔记之`GameMode`、`关卡(Level)` 和 `关卡蓝图(Level Blueprint)`的关系
  • 快速上手SElinux
  • 第8章 常用实用类
  • 基于shardingsphere的分库分表方案
  • redis读写一致问题
  • Visual Studio已更新为17.14+集成deepseek实现高效编程
  • AI大模型(二)embedding模型调用后对产生的数据进行分析
  • 水平可见直线--上凸包(andrew算法
  • 【嵙大o】C++作业合集