现在的位置: 首页 > 编程语言 > 正文

成员函数指针:(结构+与普通函数指针之间的转换)

2018年10月28日 编程语言 ⁄ 共 876字 ⁄ 字号 评论关闭

通过内存拷贝(memcpy等)可以实现任意指针

间的强制转换,但不能保证可以正常使用。

通过网上查找发现:

函数成员指针其实与普通成员指针不同,它除了包含函数本身地址以外还包含其他信息(例如是否为虚函数等),所以不能简单的理解成员函数指针就是普通指针那样一般占4字节,这个视编译器不同而不同:例如在VS中,普通成员函数指针类似于

struct  ptr{

  int * addr;

};

而虚函数的结构比较复杂,它是通过this指针加索引的方式来获取函数的真实地址,目前没有完全明白,此不赘述。

这里提获取成员函数真实地址的方法:

1 .普通成员函数

通过观察不难发现结构体的首地址就是addr的首地址,所以成员函数的入口地址其实也就是函数指针的地址,但是C++出于类型安全的考虑不允许他们转换成其他普通指针,如:

class  test {
public:
      void print(){}
};
typedef  void  (test::*cfun)();
typedef  void  (*fun)();
cfun cf = &test::print;
fun f= cf;   //失败,类型检查
memcpy(&f,&cf,sizeof(fun));
f();    //成功

2. 虚函数

(1)通过虚函数表获取

class test{
public:
       virtual void print(){}
};
typedef void  (test::*cfun)();
typedef  void  (*fun)();
test t;
int **vptr = (int**)(&t);    //vptr[0]获取虚函数表地址
cfun f = vptr[0][0];   //后面那个零时虚函数在虚函数表中的索引,表示第一个虚函数
f();
((fun) vptr[0][0])();

通常不能用&test::print获取虚函数地址,即使获取地址也是一个中间值或者总是返回0x1。

3. 通用的指针转换函数

template<class T,class R>
R  convert(T t)
{
	long addr = 0;
        memcpy(&addr,&t,sizeof(long));    
        return (R)(addr);
}

但不能保证转换的有效性。

抱歉!评论已关闭.