虚函数机制
通过分析下面的代码来了解虚函数机制
class Animal
{
public:
// 虚函数机制可以让编译器实现动态绑定
void speak()
{
cout << "动物在叫..." << endl;
}
};
未添加虚函数的类
当类中只有函数时,类的大小为1
class Animal
{
public:
// 虚函数机制可以让编译器实现动态绑定
virtual void speak()
{
cout << "动物在叫..." << endl;
}
};
当类中有虚函数时,再看看类的大小
添加虚函数之后,类的变化
当类中添加了虚函数时,类的大小变为4,同时编译器给类中添加了虚函数表指针
vfptr
这个指针指向了vftable
vftable
这张表里面存放的本类所有的虚函数入口地址
- 当发生继承时,子类会将父类的虚函数表指针继承下来,指向父类的的虚函数表,在子类调用构造函数后,编译器会将该指针指向自己的虚函数表
- 当子类重写了父类的虚函数时,在虚函数表中会用子类的函数地址去覆盖父类对应的虚函数地址
- 程序运行时,通过父类指针或者引用 调用函数时,编译器会先找到该对象中的虚函数指针,根据指针找到虚函数表,在虚函数表中找到对应的函数入口 地址
cl /d1 reportSingleClassLayout类名 类所在的文件
总结
- 当类中添加了虚函数时,类的大小变为4,同时编译器给类中添加了虚函数表指针
vfptr
这个指针指向了vftable
vftable
这张表里面存放的本类所有的虚函数入口地址- 当发生继承时,子类会将父类的虚函数表指针继承下来,指向父类的的虚函数表,在子类调用构造函数后,编译器会将该指针指向自己的虚函数表
- 当子类重写了父类的虚函数时,在虚函数表中会用子类的函数地址去覆盖父类对应的虚函数地址
- 程序运行时,通过父类指针或者引用 调用函数时,编译器会先找到该对象中的虚函数指针,根据指针找到虚函数表,在虚函数表中找到对应的函数入口 地址进行调用