C++虚函数表解析 [二]

来源:岁月联盟 编辑:exp 时间:2011-11-06

 

虚函数(Virtual Function)是通过一张虚函数表(Virtual Table)来实现的。在这个表中,主要是一个类的虚函数的地址,这张表解决了继承、覆盖的问题,其内容真实反映了实际函数。这样,在有虚函数的类的实例中虚函数表被分配在实例的内存中,所以,当用父类的指针从操作一个子类的时候,这张虚函数表就显得尤为重要了,它就像一张地图,指明了实际所应该调用的函数。

 

在C++的标准规格说明书中写道,编译器必须保证虚函数表的指针存在于对象实例中最前面的位置(这是为了保证正确取到虚函数的偏移量)。这意味着通过对象实例的地址可以得到虚函数表,然后可以遍历其中的函数指针,并调用相应的函数了。请看下面的代码

 

class Base{ 

public: 

    virtual void fun1(){cout<<"Base::fun1"<<endl;} 

    virtual void fun2(){cout<<"Base::fun2"<<endl;} 

    virtual void fun3(){cout<<"Base::fun3"<<endl;} 

private: 

    int num1; 

    int num2;  

 

}; 

typedef void(*Fun)(void); 

int main(int argc, char* argv[]) 

   Base b; 

   Fun pFun; 

   pFun=(Fun)*((int*)*(int*)(&b)+0); 

   pFun(); 

   pFun=(Fun)*((int*)*(int*)(&b)+1); 

   pFun(); 

   pFun=(Fun)*((int*)*(int*)(&b)+2); 

   pFun(); 

   return 0; 

上面程序的执行结果如下:

 

 

 

 

Base::fun1

 

Base::fun2

 

Base::fun3

 

 

可以看到通过函数指针pFun的调用,分别执行了对象b的3个虚函数。通过这个示例发现,可以强行把&b转成int*,取得虚函数表的地址,然后再次取址就可以得到第一个虚函数的地址了,也就是Base::fun1()。如果调用Base::fun2()和Base::fun3(),只需要把&b先加上数组元素的偏移,后面的步骤类似。

 

程序中Base对象b的内存结构图,如下图所示:

/

 

摘自:日新为道的专栏