每个含有虚函数的类都含有一个虚函数表,该虚函数表被所有类对象所共享。类的每个虚成员占据虚函数表中的一行。如果类中有N个虚函数,那么其虚函数表将有N*4字节的大小。虚函数(virtual function)是通过一个虚函数表(virtual table )来实现的,虚函数表里存储的是每个虚函数的地址。
虚函数表主要是一个虚函数查找表,主要是在运行时根据指针对象或者引用对象的类型调用相应的虚函数。虚函数表其实是存储了为类对象进行声明的虚函数地址。当我们创建一个类对象时,编译器会自动的生成一个指针*__vptr(一个隐藏指针),该指针指向这个类中所有虚函数的地址表。(实际上,虚函数表就是一个函数地址数组表。),请注意,*__vptr和*this指针不同,*this是一个被编译器用作解决自引用的函数参数,而*__vptr则是一个真正的指针。每一个类,不管是基类还是子类都有一个自己的virtual table,*__vptr也是被继承过来的。
class A {
public:
virtual void foo (void) {}
virtual void bar (void) {}
};
class B : public A {
public:
void foo (void) {}
};
int main(){
A* pa = new A;
pa->foo (); // A::foo
pa->bar (); // A::bar
A* pa = new B;
pa->foo (); // B::foo
pa->bar (); // A::bar
}
图解如下: