图文例解C++类的多重继承与虚拟继承

摘要:
在过去,我们总是接触到单个类的继承。然而,在现实生活中,一些新事物往往具有两种或更多事物的属性。为了解决这个问题,C++引入了多重继承的概念。C++允许您为派生类指定多个基类。这种继承结构称为多重继承。这是由于多重继承导致的继承的模糊性。以上述代码为例,如果我们希望AmphibianCar类不仅获得Vehicle的副本,而且共享Car类和Boat类的数据成员和成员函数,那么必须使用C++提供的虚拟继承技术。

在过去的学习中,我们始终接触的单个类的继承,但是在现实生活中,一些新事物往往会拥有两个或者两个以上事物的属性,为了解决这个问题,C++引入了多重继承的概念,C++允许为一个派生类指定多个基类,这样的继承结构被称做多重继承

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

图文例解C++类的多重继承与虚拟继承第1张

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

//程序作者:管宁
//站点:www.cndev-lab.com
//所有稿件均有版权,如要转载,请务必著名出处和作者
#include<iostream>
usingnamespacestd;
classVehicle
{
public:
Vehicle(intweight=0)
{
Vehicle::weight=weight;
}
voidSetWeight(intweight)
{
cout<<"重新设置重量"<<endl;
Vehicle::weight=weight;
}
virtualvoidShowMe()=0;
protected:
intweight;
};
classCar:publicVehicle//汽车
{
public:
Car(intweight=0,intaird=0):Vehicle(weight)
{
Car::aird=aird;
}
voidShowMe()
{
cout<<"我是汽车!"<<endl;
}
protected:
intaird;
};
classBoat:publicVehicle//船
{
public:
Boat(intweight=0,floattonnage=0):Vehicle(weight)
{
Boat::tonnage=tonnage;
}
voidShowMe()
{
cout<<"我是船!"<<endl;
}
protected:
floattonnage;
};
classAmphibianCar:publicCar,publicBoat//水陆两用汽车,多重继承的体现
{
public:
AmphibianCar(intweight,intaird,floattonnage)
:Vehicle(weight),Car(weight,aird),Boat(weight,tonnage)
//多重继承要注意调用基类构造函数
{
}
voidShowMe()
{
cout<<"我是水陆两用汽车!"<<endl;
}
};
intmain()
{
AmphibianCara(4,200,1.35f);//错误
a.SetWeight(3);//错误
system("pause");
}

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

2回顶部

先看如下的图示:

图文例解C++类的多重继承与虚拟继承第2张

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

由于这种模糊问题的存在同样也导致了AmphibianCar a(4,200,1.35f);执行失败,系统会产生Vehicle”不是基或成员的错误。

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

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

//程序作者:管宁
//站点:www.cndev-lab.com
//所有稿件均有版权,如要转载,请务必著名出处和作者
#include<iostream>
usingnamespacestd;
classVehicle
{
public:
Vehicle(intweight=0)
{
Vehicle::weight=weight;
cout<<"载入Vehicle类构造函数"<<endl;
}
voidSetWeight(intweight)
{
cout<<"重新设置重量"<<endl;
Vehicle::weight=weight;
}
virtualvoidShowMe()=0;
protected:
intweight;
};
classCar:virtualpublicVehicle//汽车,这里是虚拟继承
{
public:
Car(intweight=0,intaird=0):Vehicle(weight)
{
Car::aird=aird;
cout<<"载入Car类构造函数"<<endl;
}
voidShowMe()
{
cout<<"我是汽车!"<<endl;
}
protected:
intaird;
};
classBoat:virtualpublicVehicle//船,这里是虚拟继承
{
public:
Boat(intweight=0,floattonnage=0):Vehicle(weight)
{
Boat::tonnage=tonnage;
cout<<"载入Boat类构造函数"<<endl;
}
voidShowMe()
{
cout<<"我是船!"<<endl;
}
protected:
floattonnage;
};
classAmphibianCar:publicCar,publicBoat//水陆两用汽车,多重继承的体现
{
public:
AmphibianCar(intweight,intaird,floattonnage)
:Vehicle(weight),Car(weight,aird),Boat(weight,tonnage)
//多重继承要注意调用基类构造函数
{
cout<<"载入AmphibianCar类构造函数"<<endl;
}
voidShowMe()
{
cout<<"我是水陆两用汽车!"<<endl;
}
voidShowMembers()
{
cout<<"重量:"<<weight<<"顿,"<<"空气排量:"<<aird<<"CC,"<<"排水量:"<<tonnage<<"顿"<<endl;
}
};
intmain()
{
AmphibianCara(4,200,1.35f);
a.ShowMe();
a.ShowMembers();
a.SetWeight(3);
a.ShowMembers();
system("pause");
}

注意观察类构造函数的构造顺序。
虽然说虚拟继承与虚函数有一定相似的地方,但读者务必要记住,他们之间是绝对没有任何联系的!

免责声明:文章转载自《图文例解C++类的多重继承与虚拟继承》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇oc之NSSortDescriptor(描述器排序)【Maven】Maven之远程仓库的配置下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

C#3.0中的“多重继承”

    C#的对象系统是个单根系统,不支持类的多继承,只支持多接口实现,这在某种程度带来了一些不便:我们在系统设计时经常会抽象出一些接口,并为接口提供一个抽象类作为默认的实现,然后实际使用的类可以从抽象类派生。如果一个类实现了多接口,那我们只能选择一个抽象类作为祖先类,再将其他接口的实现手工加到类中。    这种情况在C#3.0中有了变化,我们现在可以利用...

python 多重继承构造函数调用顺序

实例代码 class Person(object): def __init__(self, name, age): self.name = name self.age = age print("父类构造函数") def talk(self): print("person...

设计模式之PIMPL模式

PIMPL,即private implementation的缩写,简言之就是类的声明和实现分离。 其作用概括如下: 类方法定义与函数分离,适合作为API使用 类的实现对用户来说完全是黑盒,在头文件中声明的类仅包含对用户有用的信息。 加快编译速度 a.hpp定义了类A,b.cpp调用了类A的方法。当A的方式实现变动时,传统方式b.cpp需要重新编译 PIM...