C++中的菱形继承问题

假设有一个问题,类似于鸭子这样的动物有很多种,如企鹅和鱿鱼,它们也可能会有一些共同的特性。例如,我们可以有一个叫做 AquaticBird (涉禽,水鸟的一类)的类,它又继承自 Animal 和 SwimmingAnimal,就像这样:

cpp 复制代码
class AquaticBird : public Animal, public SwimmingAnimal {  
    // AquaticBird类既是Animal也是SwimmingAnimal  
};  

如果我们同时继承了这两个类,像下图所示,将发生菱形继承:

cpp 复制代码
        Animal  
       /      \
FlyingAnimal   AquaticBird  
        \      /  
         Duck  

在这个结构中,Duck 既继承了 FlyingAnimal,也间接继承了 Animal 通过 AquaticBird 的方式。问题在于,如果我们试图调用 Animal 类的方法,例如 eat(),Duck 会造成二义性:它不清楚是通过哪个路径来调用 Animal 类的实例。

那么如何解决菱形继承问题呢

解决菱形继承问题

通过使用 虚继承,我们可以确保在最终的 Duck 类中,只有一个 Animal 的实例。

cpp 复制代码
class Animal {  
public:  
    void eat() {  
        std::cout << "Eating food." << std::endl;  
    }  
};  
class FlyingAnimal : virtual public Animal { // 使用virtual  
public:  
    void fly() {  
        std::cout << "Flying." << std::endl;  
    }  
};  
class SwimmingAnimal : virtual public Animal { // 使用virtual  
public:  
    void swim() {  
        std::cout << "Swimming." << std::endl;  
    }  
};  
class Duck : public FlyingAnimal, public SwimmingAnimal {  
    // Duck类继承自FlyingAnimal和SwimmingAnimal  
};  

现在,当 Duck 类调用 eat() 方法时,编译器知道只有一份 Animal 的实例,从而避免了二义性。

那么虚继承的底层实现原理是什么呢?

g++ -fdump -class-hlerarchy *.cpp gcc8.0之前

g++ fdump -lang - class *.cpp gcc8.0及以后

通过虚表指针偏移来实现虚继承

父类的vptr都有到共同基类的偏移量,从而让子类多继承是指向同一个"父类的父类"

只有C++有菱形继承

一般不使用菱形继承,但在库里面使用了

如下所示,下面的箭头都是继承,我们就会发现中间出现了菱形继承,iostream继承了istream和ostream。

相关推荐
爱吃KFC的大肥羊3 分钟前
C++三大特性之“继承”
开发语言·c++
毕设源码-李学长9 分钟前
计算机毕业设计java高校多媒体教室管理系统高校多媒体教室综合管理系统高校智能多媒体教室管理平台
java·开发语言·课程设计
星眸22915 分钟前
C++/QT 1
c++
先知后行。18 分钟前
线程的创建.销毁
开发语言·c++·算法
DdduZe22 分钟前
9.11作业
c++·qt
鱼嘻22 分钟前
西嘎嘎学习 - C++ 继承 - Day 10
开发语言·c++·学习·算法
孤廖28 分钟前
从 “模板” 到 “场景”,用 C++ 磨透拓扑排序的实战逻辑
开发语言·c++·程序人生·算法·贪心算法·动态规划·学习方法
我有火的意志37 分钟前
Liunx执行source /etc/profile 报错, -bash: HISTTIMEFORMAT: readonly variable
开发语言·bash·histtimeformat·readonly
老歌老听老掉牙38 分钟前
OpenCascade几何建模:平面创建与法向拉伸的工程实现
c++·平面·opencascade
-凌凌漆-1 小时前
【Qt】【C++】虚析构函数及 virtual ~Base() = default
java·c++·qt