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

1.GNU C++平台

GNU C++ 平台获取 C++ 成员虚函数地址可使用如下方法 [ 1 ] ^{[1]} [1]

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],代码如下:

#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++ 均可以成功编译。

/**********************
@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的地址
}

程序运行结果:

Base's vtable[0]:0x00C81505
Base's vtable[1]:0x00C811DB

参考文献

[1] print address of virtual member function
[2] C++动态联编实现原理分析

评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值