在过去的学习中,我们始终接触的单个类的继承,但是在现实生活中,一些新事物往往会拥有两个或者两个以上事物的属性,为了解决这个问题,C++引入了多重继承的概念,C++允许为一个派生类指定多个基类,这样的继承结构被称做多重继承。
举个例子,交通工具类可以派生出汽车和船连个子类,但拥有汽车和船共同特性水陆两用汽车就必须继承来自汽车类与船类的共同属性。
//程序作者:管宁
//站点:www.cndev-lab.com
class Vehicle
{
public:
Vehicle( int weight = 0)
{
Vehicle::weight= weight;
cout<<"载入Vehicle类构造函数"<
void SetWeight(int weight)
{
cout<<"重新设置重量"<< weight<
virtual void ShowMe() = 0;
protected:
int weight;
};
class Car: public Vehicle
{
public:
Car(int weight = 0, int aird = 0):Vehicle(weight)
{
Car::aird = aird;
cout<<"载入Car类构造函数"<
void ShowMe()
{
cout<<"我是汽车"<
protected:
int aird;
};
class Boat: public Vehicle
{
public:
Boat(int weight = 0, float tonnage = 0):Vehicle(weight)
{
Boat::tonnage = tonnage;
cout<<"载入Boat类构造函数"<
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)
{
cout<<"载入AmphibianCar类构造函数"<
void ShowMe()
{
cout<<"我是水陆两用汽车"<
void ShowMembers()
{
cout<<"重量"<
};
int main()
{
AmphibianCar a(4,200, 1.35);
a.SetWeight(3);
system("pause");
return 0;
}
上面的代码从表面看,看不出有明显的语发错误,但是它是不能够通过编译的。这有是为什么呢?
这是由于多重继承带来的继承的模糊性带来的问题。
其主要问题所在,水陆两用汽车类继承了来自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
class Vehicle
{
public:
Vehicle( int weight = 0)
{
Vehicle::weight= weight;
cout<<"载入Vehicle类构造函数"<
void SetWeight(int weight)
{
cout<<"重新设置重量"<< weight<
virtual void ShowMe() = 0;
protected:
int weight;
};
class Car:virtual public Vehicle
{
public:
Car(int weight = 0, int aird = 0):Vehicle(weight)
{
Car::aird = aird;
cout<<"载入Car类构造函数"<
void ShowMe()
{
cout<<"我是汽车"<
protected:
int aird;
};
class Boat:virtual public Vehicle
{
public:
Boat(int weight = 0, float tonnage = 0):Vehicle(weight)
{
Boat::tonnage = tonnage;
cout<<"载入Boat类构造函数"<
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)
{
cout<<"载入AmphibianCar类构造函数"<
void ShowMe()
{
cout<<"我是水陆两用汽车"<
void ShowMembers()
{
cout<<"重量"<
};
int main()
{
AmphibianCar a(4,200, 1.35);
// a.SetWeight(3);
a.ShowMe();
a.ShowMembers();
a.SetWeight(3);
a.ShowMembers();
system("pause");
return 0;
}
注意观察类构造函数的构造顺序。
【vc|c++类的多重继承与虚拟继承】
推荐阅读
- 个人日记|K8s中Pod生命周期和重启策略
- 学习分享|【C语言函数基础】
- C++|C++浇水装置问题
- 数据结构|C++技巧(用class类实现链表)
- C++|从零开始学C++之基本知识
- 步履拾级杂记|VS2019的各种使用问题及解决方法
- leetcode题解|leetcode#106. 从中序与后序遍历序列构造二叉树