`

多重继承和虚拟继承

 
阅读更多

面向对象编程语言中的多重继承指的是一个类别可以同时从多于一个父类继承行为与特征的功能。与单一继承相对,单一继承指一个类别只可以继承自一个父类。

重温Java,发现Java竟然不支持类多重继承(直接继承类),却允许接口的多重继承。C++中类可以多重继承,Java中为什么不实现这个功能呢?多重继承会带来哪些问题,从而导致Java放弃类的多重继承

再一深究,发现多年以来,多重继承一直是个敏感话题,赞成者看到的是免去笨拙的混合继承的利处,反对者看到的是多处混淆的弊端,例如变量的二义性,并且是多个变量。所以关于它的好处与风险之间孰轻孰重成为OO界多年争论的焦点。

其实最大的问题是出现拓补图,也就是出现钻石型继承结构(DOD),个人感觉这是个致命伤。正如其名:Diamond of Death。

举个简单的例子:

比如一个基类:动物。它有三个派生类:哺乳类动物,卡通类动物,宠物(确实都形成ISA关系)。现在有一个子类猫,从关系上推,它可以继承自哺乳类,卡通类,宠物,都符合ISA,如果要体现所有的特性,就需要全部继承,这样就形成了多重继承,却也形成了DOD,这样以后问题就出现了,从猫到动物的继承有三条路径,如果哺乳类,卡通类与宠物类中有相同的成员函数或变量,这样的数据组织方式会形成多义。

C++怎么解决这个问题的呢?虚继承。结果就是不得不牺牲一些内存开销,因为一个功能要在多处被重写。并且函数表里的函数指针必须调整,这样即使可以满足功能,在后期的维护也很复杂。

所以,Java才会采用这样折中的方法,硬生生的将类多重继承踢了出去。

并且,网上也有不少建议,要尽可能避免多重继承,不惜一切代价去避免钻石结构,以避免后期不可挽回的大返工。

多重继承的概念:C++允许为一个派生类指定多个基类,这样的继承结构被称做多重继承。

举个例子,交通工具类可以派生出汽车和船连个子类,但拥有汽车和船共同特性水陆两用汽车就必须继承来自汽车类与船类的共同属性。
由此我们不难想出如下的图例与代码:

clip_image001

当一个派生类要使用多重继承的时候,必须在派生类名和冒号之后列出所有基类的类名,并用逗号分隔。

//程序作者:管宁
//站点:www.cndev-lab.com
//所有稿件均有版权,如要转载,请务必著名出处和作者
#include <iostream><br>using namespace std; <br>class Vehicle <br>{ <br> public: <br> Vehicle(int weight = 0) <br> { <br> Vehicle::weight = weight; <br> } <br><b>void SetWeight(int weight) </b> <br> { <br> cout Vehicle::weight = weight; <br> } <br><b>virtual void ShowMe() = 0; <br></b> protected: <br><b>int weight; </b> <br>}; <br>class Car:public Vehicle//汽车 <br>{ <br> public: <br> Car(int weight=0,int aird=0):Vehicle(weight) <br> { <br> Car::aird = aird; <br> } <br><b>void ShowMe() </b> <br> { <br> cout } <br> protected: <br><b>int aird; <br></b>}; </iostream>

class Boat:public Vehicle//船
{
public:
Boat(int weight=0,float tonnage=0):Vehicle(weight)
{
Boat::tonnage = tonnage;
}
void ShowMe()
{
cout }
protected:
float tonnage;
};
class AmphibianCar:public Car,public Boat//水陆两用汽车,多重继承的体现
{
public:
AmphibianCar(int weight,int aird,float tonnage)
:Vehicle(weight),Car(weight,aird),Boat(weight,tonnage)
//多重继承要注意调用基类构造函数
{
}
void ShowMe()
{
cout }
};
int main()
{
AmphibianCar a(4,200,1.35f);//错误
a.SetWeight(3);//错误
system("pause");
}

上面的代码从表面看,看不出有明显的语发错误,但是它是不能够通过编译的。这有是为什么呢?
这是由于多重继承带来的继承的模糊性带来的问题。

先看如下的图示:

clip_image002

在图中深红色标记出来的地方正是主要问题所在,水陆两用汽车类继承了来自Car类与Boat类的属性与方法,Car类与Boat类同为AmphibianCar类的基类,在内存分配上AmphibianCar获得了来自两个类的SetWeight()成员函数,当我们调用a.SetWeight(3)的时候计算机不知道如何选择分别属于两个基类的被重复拥有了的类成员函数SetWeight(),导致SetWeight()的调用产生二义性。

由于这种模糊问题的存在同样也导致了AmphibianCar a(4,200,1.35f);执行失败,系统会产生Vehicle是非法的成员初始化,'Vehicle' is not a base or member。这是因为Class Car和Class Boat都继承了基类的构造函数,Vehicle()函数的调用产生二义性。

以上面的代码为例,我们要想让AmphibianCar类既获得一个Vehicle的拷贝,而且又同时拥用Car类与Boat类的数据成员与成员函数,那么就必须通过C++所提供的虚拟继承技术来实现。

我们在Car类和Boat类继承Vehicle类出,在前面加上virtual关键字就可以实现虚拟继承,使用虚拟继承后,当系统碰到多重继承的时候就会自动先加入一个Vehicle的拷贝,当再次请求一个Vehicle的拷贝的时候就会被忽略,保证继承类成员函数的唯一性。
修改后的代码如下,注意观察变化:

//程序作者:管宁
//站点:www.cndev-lab.com
//所有稿件均有版权,如要转载,请务必著名出处和作者
#include <iostream><br>using namespace std; <br>class Vehicle <br>{ <br> public: <br> Vehicle(int weight = 0) <br> { <br> Vehicle::weight = weight; <br> cout } <br><b>void SetWeight(int weight) </b> <br> { <br> cout Vehicle::weight = weight; <br> } <br> virtual void ShowMe() = 0; <br> protected: <br><b>int weight; <br></b>}; <br><font color="#ff0000">class Car:<b>virtual</b> public Vehicle</font>//汽车,这里是虚拟继承 <br>{ <br> public: <br> Car(int weight=0,int aird=0):Vehicle(weight) <br> { <br> Car::aird = aird; <br> cout } <br><b>void ShowMe()</b> <br> { <br> cout } <br> protected: <br><b>int aird; <br></b>}; <br><font color="#ff0000">class Boat:<b>virtual</b> public Vehicle//船,这里是虚拟继承</font> <br>{ <br> public: <br> Boat(int weight=0,float tonnage=0):Vehicle(weight) <br> { <br> Boat::tonnage = tonnage; <br> cout } <br><b>void ShowMe() <br></b> { <br> cout } <br> protected: <br><b>float tonnage; <br></b>}; <br><font color="#ff0000">class AmphibianCar:public Car,public Boat//水陆两用汽车,多重继承的体现 <br></font>{ <br> public: <br> AmphibianCar(int weight,int aird,float tonnage) <br> :Vehicle(weight),Car(weight,aird),Boat(weight,tonnage) <br> //多重继承要注意调用基类构造函数 <br> { <br> cout } <br><b>void ShowMe()</b> <br> { <br> cout } <br><b>void ShowMembers() <br></b> { <br> cout <p> </p> <p> } <br>}; <br>int main() <br>{ <br> AmphibianCar a(4,200,1.35f); <br> a.ShowMe(); <br> a.ShowMembers(); <br> a.SetWeight(3); <br> a.ShowMembers(); <br> system("pause"); <br>}</p> <p>注意观察类构造函数的构造顺序。 <br><b>虽然说虚拟继承与虚函数有一定相似的地方,但读者务必要记住,他们之间是绝对没有任何联系的!</b></p> <p>补充:</p> <p>1、 当一个类有多个父类时,每个父类在内存中依次排列,然后该类自己的成员。 <br>2、 每一个父类的镜像中,都包含有独立的虚函数表 <br>3、 当把子类的指针Upcast的时候,两种Upcast的方式得到的结果分别指向各自的父类镜像 <br>4、 当两个父类重载的虚函数不同时,会使用Thunk机制,也就是说,虚函数表中的函数指针并不指向实际的虚函数,而是指向一小段代码。在这一小段代码中,会修改This指针(ECX寄存器),使之指向合适的父类镜像,然后再跳转到实际的虚函数体。 <br>5、 当不使用虚继承时,共同基类的成员对象,在子类中会有独立两分(从两个父类各自继承了一份)。 <br>6、 当使用虚继承时,共同基类的成员对象也会在虚函数表中记录,访问它必须先查找虚函数表。</p> </iostream>

分享到:
评论

相关推荐

    C++ 多重继承和虚拟继承对象模型、效率分析

    C++的多重继承和虚拟继承是面向对象编程中两种高级特性,它们允许类从多个基类继承特性,但同时也引入了复杂性和效率问题。本文主要探讨了这两种继承方式的基本概念,对象模型的差异,以及它们对运行效率的影响。 ...

    C++虚拟多重继承对象模型讨论-样例

    这种继承方式的主要目的是解决“菱形问题”(Diamond Problem),即在非虚拟继承的多重继承中,派生类可能会有多于一份基类子对象,导致二义性。本篇讨论将深入探讨虚拟多重继承的对象模型,以及如何通过样例程序来...

    多重继承(6_7钻石继承问题)_C++_

    然而,多重继承也带来了一些复杂性和潜在的问题,其中最著名的就是所谓的“钻石继承”问题。在本篇中,我们将深入探讨多重继承以及如何解决钻石继承问题。 首先,让我们理解多重继承的基本概念。假设我们有三个类A...

    实现c++类的多重继承的例程

    在C++编程语言中,多重继承是一种特性,允许一个派生类从多个基类中继承属性和行为。这种设计模式增强了代码的复用性,并且能够构建复杂的类层次结构。以下是一个关于如何实现C++类的多重继承的详细解释。 首先,让...

    多重继承及虚继承中对象内存的分布

    尽管在理想的使用环境中,一个C++程序员并不需要了解这些编译器内部实现细节,实际上,编译器针对多重继承(特别是虚拟继承)的各种实现细节对于我们编写C++代码都或多或少产生一些影响(比如downcastingpointer、...

    07-08面向对象程序设计及答案A.doc

    C++中有三种继承方式:单继承、多重继承和虚拟继承。单继承是指派生类只有一个基类;多重继承是指派生类可以从多个基类继承;虚拟继承主要是为了解决多重继承中“菱形问题”,确保基类的子对象只有一份,避免内存...

    C 程序设计课件:第17章 多重继承.ppt

    C++ 程序设计课件:多重继承 多重继承是 C++ 编程语言中的一种继承机制,它允许一个类继承多个基类,从而使得子类拥有多个基类的...但是,多继承也存在一些问题,如模糊性问题,可以通过虚拟继承和访问控制来解决。

    C++面向对象技术完全剖析_源代码(继承,封装,多态,虚函数,纯虚函数,虚拟继承,多重继承,函数重载,指针……)

    3、体现继承 虚拟继承(要通过至少三层 父类父类子类) 虚函数 (3层 纵向关系) 水平方向上:体现出继承顺序 先虚拟继承 再普通继承 通过实例化类 体现对象构造和析构的顺序 还要有函数重载 指针 指针悬挂 无参...

    2005年C++程序设计期终考试试题.doc

    虚拟继承是在多重继承中引入的一个概念,它可以解决“菱形问题”,即当一个类同时继承了具有相同成员的两个父类时,通过虚拟继承可以确保只有一个副本。 3. 带有纯虚函数的类被称为抽象类。抽象类不能被实例化,...

    C++程序设计课件:第17章 多重继承.ppt

    在C++编程语言中,多重继承是一种特性,允许一个派生类从多个基类中继承属性和行为。这种机制在第17章“多重继承”中被深入探讨。以下是关于多重继承及其相关概念的详细解释: 17.1 多继承如何工作: 多继承意味着...

    C++复习题及参考答案.docx

    - C++支持单继承、多重继承和虚拟继承。继承允许一个类(派生类)从另一个类(基类)继承属性和行为,从而实现代码复用和类层次结构的构建。 9. **编译和链接过程**: - 源代码首先通过编辑器编写,然后由编译器...

    C++(包含了C++的各种代码)

    C++支持单继承、多重继承和虚拟继承。 4. **多态性**:多态性是指同一消息可以根据接收者的不同而表现出不同的行为。在C++中,多态性有两种形式:静态多态(编译时多态)和动态多态(运行时多态)。静态多态主要...

    面向对象程序设计课件(C++)

    C++支持单继承、多重继承和虚拟继承,其中虚拟继承可以解决多继承中的二义性问题。 4. **多态性**:多态性使得不同对象能对同一消息作出不同的响应,分为静态多态(编译时多态,如函数重载、运算符重载)和动态多态...

    大学VC课件

    - C++支持单继承、多重继承和虚拟继承,其中虚拟继承可以解决多继承中的二义性问题。 6. **访问控制(Access Control)**: - C++有三种访问修饰符:public、private和protected。它们控制类的成员对其他类的可见...

    c++ 类基本概念易错点

    - 继承有单继承、多重继承和虚拟继承等形式,多重继承可能导致菱形问题,虚拟继承可以解决。 理解并掌握这些知识点对于编写高质量的C++代码至关重要。在实际编程中,应注意避免这些易错点,确保代码的正确性和可...

    C++学习 PPT《明日科技丛书》

    C++支持三种继承方式:单一继承、多重继承和虚拟继承,增强了代码的重用性和灵活性。 2. **构造函数与析构函数**:构造函数是在创建对象时自动调用的特殊函数,用于初始化对象的状态;析构函数则在对象生命周期结束...

    面向对象课程

    C++中的继承有单继承、多重继承和虚拟继承等。 4. **多态**:多态是指同一消息可以根据发送对象的不同而采取多种不同的行为方式。C++通过虚函数和纯虚函数实现多态,允许子类重写父类的函数,实现动态绑定。 5. **...

    2019 面向对象程序设计(C++)考试要求1

    10. **继承与多态**:熟悉单一继承、多重继承和虚拟继承,掌握虚函数实现的多态性,以及其在软件设计中的复用性价值。 11. **异常处理**:了解如何在C++中抛出和捕获异常,以及异常安全编程的重要性。 12. **模板*...

Global site tag (gtag.js) - Google Analytics