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

server/2024/11/28 15:09:33/
  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.ppmy.cn/server/145655.html

相关文章

python读txt文件时出现UnicodeDecodeError错误的解决

1 现象 在编写文件读的代码&#xff1a; src_file_path "a:\\src.txt" with open(src_file_path) as file:data file.readline()出现如下错误&#xff1a; > UnicodeDecodeError: gbk codec cant decode byte 0xab in position 2: illegal multibyte sequenc…

【网络安全设备系列】12、态势感知

0x00 定义&#xff1a; 态势感知&#xff08;Situation Awareness&#xff0c;SA&#xff09;能够检测出超过20大类的云上安全风险&#xff0c;包括DDoS攻击、暴力破解、Web攻击、后门木马、僵尸主机、异常行为、漏洞攻击、命令与控制等。利用大数据分析技术&#xff0c;态势感…

线索二叉树

1.什么是线索二叉树&#xff1f; 线索二叉树是一种特殊的二叉树&#xff0c;它在传统二叉树的基础上&#xff0c;利用节点中原本为空的指针域存储指向节点前驱或后继的信息&#xff0c;从而在遍历时不需要递归或栈辅助&#xff0c;能够高效地找到前驱或后继节点。 前序遍历&am…

代码随想录打卡DAY20

算法记录第20天 [二叉树] 1.LeetCode 501. 二叉搜索树中的众数 题目描述&#xff1a; 给你一个含重复值的二叉搜索树&#xff08;BST&#xff09;的根节点 root &#xff0c;找出并返回 BST 中的所有 众数&#xff08;即&#xff0c;出现频率最高的元素&#xff09;。如果树…

深入解析经典排序算法:原理、实现与优化

文章目录 6. 堆排序&#xff08;Heap Sort&#xff09;原理Java 实现 7. 希尔排序&#xff08;Shell Sort&#xff09;原理Java 实现 8. 计数排序&#xff08;Counting Sort&#xff09;原理Java 实现 9. 基数排序&#xff08;Radix Sort&#xff09;原理Java 实现 深入浅出&am…

(STM32)ADC驱动配置

1.ADC驱动&#xff08;STM32&#xff09; ADC模块中&#xff0c;**常规模式&#xff08;Regular Mode&#xff09;和注入模式&#xff08;Injected Mode&#xff09;**是两种不同的ADC工作模式 常规模式&#xff1a;用于普通的ADC转换&#xff0c;是默认的ADC工作模式。 注入…

【ROS2】ROS2 与 ROS1 编码方式对比(C++实现)

目录 一、初始化和关闭节点二、发布者和订阅者三、服务和客户端四、参数管理五、日志记录六、生命周期管理 ROS1 主要使用roscpp和rospy作为客户端库&#xff0c;分别用于C和Python语言。 ROS2 引入了新的客户端库rclcpp&#xff08;C&#xff09;和rclpy&#xff08;Python&a…

iOS开发之修改已有项目的项目名和类名前缀

一、修改项目名称 1、Xcode打开项目修改项目名称 直接选中项目&#xff0c;点击enter&#xff0c;直接修改项目名称 把buydodo改成xiedodo&#xff0c;点击enter Rename完了点继续&#xff0c;只有框框内的部分变了 2、退出Xcode关闭项目&#xff0c;修改剩下的项目名称 找…