先看示例代码:
#include <iostream.h>
class person
{
public:
virtual void name()
{
cout<<"A::name"<<endl;
}
private:
virtual void sex()
{
cout<<"A::sex"<<endl;
}
};
class student:public person
{
public:
virtual void name()
{
cout<<"B::name"<<endl;
}
virtual void address()
{
cout<<"B::address"<<endl;
}
private:
virtual void ID()
{
cout<<"B::ID"<<endl;
}
};
typedef void (*Fun)(void);//定义一个函数指针
void main()
{
student stu;
for(int i=0;i<4;i++)
{
Fun p = (Fun)*((int*) * (int*)(&stu)+i);
p();
}
}
运行结果:
说明:
首先在for循环内,使用函数指针获得Student类对象stu的首地址,因为stu的首地址保存了一个指针,这个指针指向了这个类的虚函数表,而这个虚函数表包含了父类person的虚函数表,当子类覆盖了父类中的虚函数时(此例子中的virtual void name()),那么std的虚函数表中原本指向父类的虚函数name()就改指向子类stu的name()了。
其中要注意的是,父类的虚函数表的函数顺序是按照申明的顺序并且父类的虚函数在子类的虚函数前面。因此对象stu的首地址指向的就是被自己覆盖的name()函数了。
A类的虚函数表是这样的:
A::name() |
A::sex() |
B类的虚函数表示这样的:
B::name() ->原本是A::name() |
A::sex() |
B::address() |
B::ID() |