深度探索c++对象模型-06-多重继承函数表分析

本文探讨了C++中的多继承和虚函数表的实现细节。通过一个示例代码展示了如何在派生类中处理来自不同基类的虚函数,并解释了对象内存布局中虚函数表指针的存放方式。分析表明,每个基类都有自己的虚函数表,而子类可能共享或拥有额外的虚函数表,以处理继承自不同基类的虚函数。
摘要由CSDN通过智能技术生成

先看代码分析:

#include <iostream>
using namespace std;
class base1
{
public: 
	virtual void f()
	{
		cout << "base1::f()\n";

	}
	virtual void g()
	{
		cout << "base1::g()\n";
	}
};


class base2
{
public: 
	virtual void h()
	{
		cout << "base2::h()\n";
	}

	virtual void i()
	{
		cout << "base2::i()\n";
	}
};

class derived:public base1,public base2
{
	virtual void f()
	{
		cout << "derived::f()\n";
	}

	virtual void i()
	{
		cout << "derived::i()\n";
	}

	virtual void x()
	{
		cout << "derived::x()\n";
	}

};

typedef void(*Func)();
void main()
{
	cout << "sizeof(base1):" << sizeof(base1) << endl;
	cout << "sizeof(base2):" << sizeof(base1) << endl;
	cout << "sizeof(derived):" << sizeof(derived) << endl;
	cout << "----------------------------\n";
	derived d;
	base1 &b1 = d;
	base2 &b2 = d;
	b1.f();
	b1.g();

	b2.h();
	b2.i();
	cout << "------------------------------------\n";
	long *vptr_addr1 = (long *)&d;
	long *vptr1 = (long *)(*vptr_addr1);//这是vptr1的值,也就是虚函数表的地址(第一个虚函数表指针所指的虚表地址)
	Func f1 = (Func)vptr1[0];
	f1();


	long *vptr_addr2 = vptr_addr1 + 1;

	long *vptr2 = (long *)(*vptr_addr2);

	Func x = (Func)vptr2[0];
	x();

	Func y = (Func)vptr2[1];
	y();

	system("pause");
}

结果:
在这里插入图片描述
总结:
从运行结果可以看出子类是8个字节,所以它有两个虚函数表指针。
一个对象,如果它的类有多个基类则有多个虚函数表指针(注意是两个虚函数表指针,而不是两个虚函数表)
在多继承中,对应各个基类的vptr按继承顺序依次放置在类的内存空间中,且子类与第一个基类共用一个vptr(第二个基类有自己的vptr)
在这里插入图片描述
从图可以看出:
(1)子类对象有两个虚函数表指针vptr1、vptr2;
(2)同时子类也有两个虚函数表,因为它继承自两个基类。
(3)子类和第一个基类共用一个vptr(因为指向一个虚函数表,所以也可以说子类和第一个基类共用一个虚函数表vtbl)因为从图可以看出vtbl1有5个虚函数,其余三个是子类自己的。

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

发如雪-ty

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值