菱形继承:

wKiom1cFDbHDMc8eAAAx4qLLT30181.png

Assitant的菱形继承对象模型

wKioL1cFEW_C0hdOAAAoQTE86eQ859.png

Assitant中有两份Person成员,存在二义性和数据冗余。

所以我们引出了虚拟继承。

virtual虚继.不会在子类中创建父类成员,但是子类中可以引用,就像指针一样。主要用在菱形继承,也叫钻石继承。

虚拟继承对象模型

wKioL1cFElbiO1BmAAAtbIPfI_w316.png

class Student : vitrual public Person
class Teacher : virtual public Peraon

虚函数表

  通过一块连续内存来存储虚函数的地址。这张表解决了继承、虚函数(重写)
问题。在有虚函数的对象实例中都存在一张虚函数表,虚函数表就像一张地图,指明了实
际应该调用的虚函数。

举例:

typedef  void(*PFUN)();
void PrintVT(int p)//打印虚函数表的信息
{
	PFUN pfun = NULL;
	int *ptr = (int *)p;
	cout << "虚表地址->" << ptr << endl;
	int i = 0;
	while (ptr[i])
	{
		pfun = (PFUN)ptr[i];
		cout << "第" << i << "个虚函数地址-》 " << pfun<<"----";
		pfun();
		i++;
	}
}
class B
{
public:
	virtual void fun1()
	{
		cout << "B::fun1()" << endl;
	}
	virtual void fun2()
	{
		cout << "B::fun2()" << endl;
	}

private:
	int _b;
};

class D :public B
{
public:
	virtual void fun1()
	{
		cout << "D::fun1()" << endl;
	}
	virtual void fun3()
	{
		cout << "D::fun3()" << endl;
	}
	virtual void fun4()
	{
		cout << "D::fun4()" << endl;
	}
private:
	int _d;
};

int main()
{
	D d;
	B b;
	PrintVT(*(int *)&d);
	cout << endl;
	PrintVT(*(int *)&b);
	getchar();
	return 0;
}

wKioL1cGJFSyN1D7AAAgBUZRhBo698.png


其结果如下:

wKioL1cGI8GA-7u0AAAmkz8_NZs087.png


这就是我们所说的虚函数。因为在D里重写的B::fun1();所以D中的fun1()就被覆盖了。小伙伴们懂了吗?j_0037.gif