1.GNU C 平台
GNU C 平台获取C 成员虚函数地址可使用如下方法[1]^{[1]}[1]:
代码语言:javascript复制class Base
{
int i;
public:
virtual void f1()
{
cout<<"Base's f1()"<<endl;
}
};
Base b;
void (Base::*mfp)() = &Base::f1;
printf("address: %p", (void*)(b->*mfp));
上面的代码通过GCC 4.8.5中编译通过。
2.Visual C 平台
可以采用内联汇编的方式获取[2]^{[2]}[2],代码如下:
代码语言:javascript复制#define ShowFuncAddress(function) _asm{
mov eax, function}
_asm{mov p,eax}
cout<<"Address of "#function": "<<p<<endl;
//使用示例
ShowFuncAddress(Base::f1);
上面的代码在VS2017中编译通过。
3.通过访问虚函数表获取虚函数地址
下面的代码可以在GCC和Visual C 共同编译运行。
代码语言:javascript复制/**********************
@className:类名称
@pObj:类对象地址
@index:虚函数表项(从0开始)
**********************/
void showVtableContent(char* className, void* pObj, int index)
{
unsigned long* pAddr=NULL;
pAddr=reinterpret_cast<unsigned long*>(pObj);
pAddr=(unsigned long*)*pAddr; //获取虚函数表指针
cout<<className<<"'s vtable["<<index<<"]:0x"<<(void*)pAddr[index]<<endl;
}
//使用示例:
class Base
{
int i;
public:
virtual void f1()
{
cout<<"Base's f1()"<<endl;
}
virtual void f2()
{
cout<<"Base's f2()"<<endl;
}
};
int main()
{
Base b;
showVtableContent("Base",&b,0); //输出第一个虚函数Base::f1的地址
showVtableContent("Base",&b,1); //输出第二个虚函数Base::f2的地址
}
程序运行结果:
代码语言:javascript复制Base's vtable[0]:0x00C81505
Base's vtable[1]:0x00C811DB
参考文献
[1]print address of virtual member function [2]C 动态联编实现原理分析