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

c++的虚继承说明、案例、代码

  1. 虚继承的基本概念

    • 在 C++ 中,虚继承主要用于解决多继承时可能出现的菱形继承问题。菱形继承是指一个类有两个(或更多)子类,而这两个子类又同时继承自一个共同的基类,当这些子类又被另一个类继承时,就形成了菱形结构。在这种情况下,如果没有虚继承,会导致基类数据成员在派生类中有多份副本,可能引起二义性等问题。虚继承可以保证在这种复杂的继承关系中,公共基类只有一份副本。
  2. 简单的虚继承示例

    • 首先看一个没有虚继承导致数据成员重复的例子:
    • cpp
class Base {
public:int baseData;
};class Derived1 : public Base {
};class Derived2 : public Base {
};class GrandDerived : public Derived1, public Derived2 {
};int main() {GrandDerived gd;// 下面这行代码会产生二义性错误,因为baseData在Derived1和Derived2中都存在// gd.baseData = 10;return 0;
}

  • 在这个例子中,GrandDerived类通过Derived1Derived2间接继承了Base类,这就导致GrandDerived对象中有两份Base类的数据成员baseData。当试图访问baseData时会产生二义性错误。

  1. 使用虚继承解决菱形继承问题

    • 下面是使用虚继承来解决上述问题的代码
    • cpp
class Base {
public:int baseData;
};class Derived1 : virtual public Base {
};class Derived2 : virtual public Base {
};class GrandDerived : public Derived1, public Derived2 {
};int main() {GrandDerived gd;gd.baseData = 10;  // 正确,此时只有一份baseDatareturn 0;
}
  • 在这个修改后的代码中,Derived1Derived2虚继承自Base类。这使得在GrandDerived类中,Base类只会有一份副本,所以可以正确地访问baseData成员。

  1. 虚继承的构造函数顺序案例

    • 当涉及虚继承时,构造函数的调用顺序也有特殊的规则。构造函数的调用顺序是先调用虚基类的构造函数,然后再按照继承顺序调用非虚基类的构造函数。
    • cpp
class Base {
public:Base() {std::cout << "Base constructor" << std::endl;}
};class Derived1 : virtual public Base {
public:Derived1() {std::cout << "Derived1 constructor" << std::endl;}
};class Derived2 : virtual public Base {
public:Derived2() {std::cout << "Derived2 constructor" << std::endl;}
};class GrandDerived : public Derived1, public Derived2 {
public:GrandDerived() {std::cout << "GrandDerived constructor" << std::endl;}
};int main() {GrandDerived gd;return 0;
}
  • 在这个例子中,输出结果是:
Base constructor
Derived1 constructor
Derived2 constructor
GrandDerived constructor
  • 可以看到,首先调用了虚基类Base的构造函数,然后按照继承顺序调用了Derived1Derived2的构造函数,最后调用了GrandDerived的构造函数。

  1. 虚继承中的指针和引用案例

    • 考虑以下代码来展示虚继承中指针和引用的行为:
    • cpp
class Base {
public:int baseData;virtual void print() {std::cout << "Base print" << std::endl;}
};class Derived1 : virtual public Base {
public:void print() override {std::cout << "Derived1 print" << std::endl;}
};class Derived2 : virtual public Base {
public:void print() override {std::cout << "Derived2 print" << std::endl;}
};class GrandDerived : public Derived1, public Derived2 {
};int main() {GrandDerived gd;Base* ptr = &gd;ptr->print();  // 调用Derived1的print函数,这取决于继承顺序和虚函数机制return 0;
}
  • 在这个例子中,通过Base*指针指向GrandDerived对象,当调用print函数时,由于虚函数的动态绑定特性和继承顺序,实际上调用的是Derived1类中的print函数。这展示了在虚继承场景下,通过基类指针或引用访问虚函数时的多态行为。

以下是用流程图来说明虚继承用于解决多继承时菱形继承问题的过程:

graph TD;A[定义基类Base] --> B[定义子类Derived1和Derived2直接继承Base];B --> C[定义GrandDerived类继承Derived1和Derived2形成菱形继承结构];C --> D[不使用虚继承时,GrandDerived对象中有两份Base类的数据成员,访问可能出现二义性];A --> E[定义子类Derived1和Derived2虚继承Base];E --> F[定义GrandDerived类继承Derived1和Derived2];F --> G[使用虚继承后,Base类在GrandDerived对象中只有一份副本,可正常访问数据成员];

在上述流程图中:

  • 首先是定义一个基类Base
  • 然后有两种情况分支:
    • 一种是常规的非虚继承方式,Derived1Derived2直接继承Base,之后GrandDerived再继承Derived1Derived2,这样会形成菱形继承结构,并且在不使用虚继承时,GrandDerived对象中会存在两份Base类的数据成员,导致在访问这些数据成员时可能出现二义性问题。
    • 另一种是采用虚继承的方式,Derived1Derived2虚继承Base,接着GrandDerived继承Derived1Derived2,此时由于虚继承的作用,Base类在GrandDerived对象中只会有一份副本,从而可以正常地访问数据成员,避免了二义性等问题。
http://www.lryc.cn/news/492824.html

相关文章:

  • 小米PC电脑手机互联互通,小米妙享,小米电脑管家,老款小米笔记本怎么使用,其他品牌笔记本怎么使用,一分钟教会你
  • 介绍SSD硬盘
  • CMAKE常用命令详解
  • Vue3的通灵之术Teleport
  • ue5第三人称闯关游戏学习(一)
  • IIC 随机写+多次写 可以控制写几次
  • controller中的参数注解@Param @RequestParam和@RequestBody的不同
  • 手搓人工智能-最优化算法(1)最速梯度下降法,及推导过程
  • 多目标优化算法——多目标粒子群优化算法(MOPSO)
  • Swift——自动引用计数ARC
  • 【Quarkus】基于CDI和拦截器实现AOP功能(进阶版)
  • 【踩坑日记】【教程】如何在ubuntu服务器上配置公钥登录以及bug解决
  • insmod一个ko提供基础函数供后insmod的ko使用的方法
  • 七、传统循环神经网络(RNN)
  • LeetCode:19.删除链表倒数第N个节点
  • 【RISC-V CPU debug 专栏 2 -- Debug Module (DM), non-ISA】
  • 单片机学习笔记 11. 外部中断
  • 基于stm32的智能教室管理系统/智能家居系统
  • 基于 Qt 和 GStreamer 的环境中构建播放器
  • windows docker 入门
  • baomidou Mabatis plus引入异常
  • 深度学习中的正则化模型是什么意思?
  • 修改IDEA配置导致Spring Boot项目读取application.properties中文乱码问题
  • Flink 热存储维表 使用 Guava Cache 减轻访问压力
  • 深入探索SenseVoiceSmall:高效多语言语音识别与处理模型
  • Flink--API 之Transformation-转换算子的使用解析
  • 每日十题八股-2024年11月27日
  • OpenCV截取指定图片区域
  • Java部分新特性
  • 【SpringBoot】28 API接口防刷(Redis + 拦截器)