虚继承与虚函数

虚继承
当我们学习了继承以后,我们会碰到这样一个问题。
A派生出B和C,而B和C一起派生出了D,这时候D中会有两份一样的数据成员与成员函数,如图:

在这里插入图片描述
例:

class Base
{
public:
	void fn()
	{
		cout<<"this is Base::fn()"<<endl;	
	}
};
class Base1:public Base
{
};
class Base2:public Base
{
};
class Derived:public Base1,public Base2
{
};

这时候因为Base1从Base里继承了一个fn(),Base2从Base里也继承了一个fn(),而Base1和Base2共同派生出一个Derived,它里面现在就会有两个fn(),一个是继承Base1的,一个是继承Base2的,这时候我们定义一个Derived对象的话,直接调用fn()会导致不知道调用哪一个。

为了解决这样的问题,我们学习了虚继承(virtual)。
如图:
在这里插入图片描述
这里我们对class A进行修饰就不会有这种情况发生了。

注意:
在派生类执行构造函数时,先调用虚基类的构造函数。

例: 床,沙发,沙发床

#include <iostream>
using namespace std;

class Furniture {
public:
	Furniture(int d = 100) :data(d) {
		cout << "Furniture()" << data << endl;
	}
	int data;
};

class Sofa :virtual public Furniture {
public:
	Sofa() :Furniture() {
		cout << "Sofa()" << data << endl;
	}
	void setD(int d)
	{
		data = d;
	}
};

class Bed :virtual public Furniture {
public:
	Bed() :Furniture() {
		cout << "Bed()" << data << endl;
	}
	int getD()
	{
		return data;
	}
};

class SofaBed :public Sofa, public Bed {
public:
	SofaBed() :Sofa(), Bed(), Furniture(100) {
		cout << "C()" << data << endl;
	}
	void dis() {
		cout << data << endl;
		cout << Sofa::data << endl;
		cout << Bed::data << endl;
	}
};

int main() {
	SofaBed c;
	c.dis();
	c.setD(10000);
	c.setD(2000);
	cout << c.getD() << endl;
	c.dis();
}

更改床、沙发、沙发床的长宽高它们的不同状态的长宽高都会发生变化。

虚函数
在我们实现类的多态时,会定义一个抽象类和好多个实例类
他们其中会有相同的函数,但是实现的功能不能相同,这时候我们就要用到虚函数。

虚函数多态:
1.至少有两个有继承关系的类;
2.基类与派生类之间有同名同参的函数,且基类函数被virtual修饰(覆盖)
3.必须通过基类的引用或指针调用该函数。

说虚函数之前说说重载、覆盖与隐藏的概念:
重载:在同一作用域中,同名不同参的函数叫重载
覆盖:分别位于基类和派生类之中,同名同参且基类函数被virtual所修饰
隐藏:分别位于基类和派生类中
a.同名同参积累函数未被virtual所修饰
b.同名不同参

例如:

class Stuff  //职工
{
public:
	virtual void work()=0;
};
class Coder:public Stuff		//码农
{
public:
	void work()
	{
		cout<<"编码"<<endl;
	}
};
class Driver:public Stuff		//司机
{
public:
	void work()
	{
		cout<<"开车"<<endl;
	}
};

int main()
{
	Coder c;
	Stuff* p=&c;
	p->work();
	return 0;
}

这时候输出的就是“编码”,这就是虚函数的作用,而例子中写的 virtual void work()=0;是一个纯虚函数,他所在的Stuff类就是一个抽象类。

虚函数写法:
virtual 函数类型 函数名 (形参表)

注:
如果基类函数为虚函数,则其派生类中同名同参的函数即使没有virtual修饰也是虚函数(一虚到底)。

虚函数实现原理:
虚函数表(虚表):是一个存放本类中所以虚函数地址的静态指针数组。
在类中如果有虚函数,则会有一个隐藏的成员指针变量来指向虚表,所以有虚函数的类会比多出四个字节来存放指针。

实现过程:
在这里插入图片描述

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值