理论知识:
当类中声明虚函数时,编译器会在类中生成一个虚函数表
虚函数表是一个存储类成员函数指针的数据结构
虚函数表是由编译器自动生成与维护的
virtual成员函数会被编译器放入虚函数表中
当存在虚函数时,每个对象中都有一个指向虚函数表的指针(C++编译器给父类对象、子类对象提前布局vptr指针;当执行函数是,C++编译器不需要区分子类对象或者父类对象,只需要在base指针中,找vptr指针即可。)
VPTR一般作为类对象的第一个成员
多态的实现原理
C++中多态的实现原理
当类中声明虚函数时,编译器会在类中生成一个虚函数表
虚函数表是一个存储类成员函数指针的数据结构
虚函数表是由编译器自动生成与维护的
virtual成员函数会被编译器放入虚函数表中
存在虚函数时,每个对象中都有一个指向虚函数表的指针(vptr指针)
1 | class parent |
parent对象:
1 | VTABLE |
child对象
1 | VTABLE |
1 | void run(parent *p) |
说明1:
通过虚函数表指针VPTR调用重写函数是在程序运行时进行的,因此需要通过寻址操作才能确定真正应该调用的函数。而普通成员函数是在编译时就确定了调用的函数。在效率上,虚函数的效率要低很多。
说明2:
出于效率考虑,没有必要将所有成员函数都声明为虚函数
说明3 :C++编译器,执行函数,不需要区分是子类对象还是父类对象
如果p调用的函数是虚函数,就去找p,再找vptr,虚函数表,执行函数,所以C++编译器不需要知道是子类对象还是父类对象,给我们的程序员造成的假象是C++编译器能识别子类对象或者父类对象。
如果是普通函数,就根据类型,在静态链编的时候,就确定要调用的函数。
如何证明vptr指针的存在
1 |
|
构造函数中能调用虚函数,实现多态吗
对象中的VPTR指针什么时候被初始化?
对象在创建的时,由编译器对VPTR指针进行初始化
只有当对象的构造完全结束后VPTR的指向才最终确定
父类对象的VPTR指向父类虚函数表
子类对象的VPTR指向子类虚函数表
要初始化一个子类的vptr指针,这个初始化时分步骤的:
当执行父类的构造函数时,子类对象的vptr指向的是父类的虚函数表,当父类的构造函数运行完毕后,会把vptr指向子类的虚函数表。
1 |
|
虚析构函数
1 |
|