虚函实现原理

对于含有虚函数的类,编译器会自动为其生成一个一维的虚函数表vtable,用来存储虚函数的入口地址。然后在实例化类对象的时候,会自动创建一个指向该类虚函数表的指针vptr。

       当一个子类继承父类(含虚函数)时,会赋值一个相同的虚函数表。然后如果子类中对父类中的虚函数进行重写(注意:两个函数的函数名、函数标签都相同时才算同一个函数),那么就会再开辟一片空间来存储重写过的函数,子类的vtable中相应的入口地址也会被修改成这个重写函数的入口地址。

       

class A
{
public:
	int a,b;
	void fun1(){...};
        void fun2(){...};
	virtual void virfun1(){...};// 假设存储该函数的地址为1
	virtual void virfun2(){...};//假设存储该函数的地址为2
};
class B public:A
{
public:
        int c;
	virtual void virfun1(){xxxxxx};//假设存储该函数的地址为3.
};

<span style="font-family: Arial, Helvetica, sans-serif; background-color: rgb(255, 255, 255);">        在某一类的具体实例中,内存中只保存了该实例的状态变量(成员变量)以及一个指向该类虚函数表的指针vptr,而没有成员函数。对于非虚函数,是在编译时编译器自动为成员函数绑定入口地址的,而虚函数则要在运行时动态绑定(查水表!。。好吧,是虚函数表。。)</span>

       上面一段代码中,A类中,假设virfun1对应的地址为1,virfun2对应的虚函数表为2, 则 虚函数应该是 1,2。 然后B类, 由于重写了virfun1(假设地址为3),其虚函数表应该是3,2。 


     另外,在虚函数通常是方便指针变量的使用的, 比如:

   A* p = new B;
     这个在下一篇文章 ,虚函数&继承 中会说到。。;

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章