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

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;  // 正确,此时只有一份baseData
    return 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.kler.cn/a/412982.html

相关文章:

  • 【Kubernetes 指南】基础入门——Kubernetes 简介(一)
  • 攸信技术:运动文化激发企业活力,赋能体育行业新未来
  • Uniapp开发下拉刷新功能onPullDownRefresh/onReachBottom
  • 如何选择最适合企业的ETL解决方案?
  • 34 基于单片机的指纹打卡系统
  • 设计模式——业务代表模式
  • 网络药理学之薛定谔Schrödinge Maestro:6、分子对接(Glide、Ligand docking)和可视化
  • 【人工智能】Python常用库-TensorFlow常用方法教程
  • C语言编译和链接讲解
  • 【k8s深入学习之 Scheme】全面理解 Scheme 的注册机制、内外部版本、自动转换函数、默认填充函数、Options等机制
  • RocketMQ: 消息过滤,通信组件,服务发现
  • 探索Python WebSocket新境界:picows库揭秘
  • 哈希表理解与底层模拟实现
  • Python的排序算法
  • 深度学习创新点不足?试试贝叶斯神经网络!
  • Python中的DrissionPage详解
  • Rust eyre 错误处理实战教程
  • 针对静态交通停车诱导系统解决方案及停车开源框架实现
  • 目录遍历漏洞-CVE-2021-41773
  • C#基础31-35
  • 极狐GitLab 17.6 正式发布几十项与 DevSecOps 相关的功能【一】
  • 『VUE』elementUI dialog的子组件created生命周期不刷新(详细图文注释)
  • 【go】查询某个依赖是否存在于这个代理
  • 【Python TensorFlow】进阶指南(续篇四)
  • 写一个流程,前面的圆点和线,第一个圆上面没有线,最后一个圆下面没有线
  • 初识java(3)