C++获取类成员虚函数地址

2019-02-22 11:25:48 浏览数 (1)

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 动态联编实现原理分析

0 人点赞