成员函数指针的结构以及怎么与普通函数指针之间的转换-创新互联

这篇文章给大家介绍成员函数指针的结构以及怎么与普通函数指针之间的转换,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。

让客户满意是我们工作的目标,不断超越客户的期望值来自于我们对这个行业的热爱。我们立志把好的技术通过有效、简单的方式提供给客户,将通过不懈努力成为客户在信息化领域值得信任、有价值的长期合作伙伴,公司提供的服务项目有:域名与空间、网页空间、营销软件、网站建设、垦利网站维护、网站推广。

通过内存拷贝(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
R  convert(T t)
{
	long addr = 0;
        memcpy(&addr,&t,sizeof(long));    
        return (R)(addr);
}

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

关于成员函数指针的结构以及怎么与普通函数指针之间的转换就分享到这里了,希望以上内容可以对大家有一定的帮助,可以学到更多知识。如果觉得文章不错,可以把它分享出去让更多的人看到。


本文标题:成员函数指针的结构以及怎么与普通函数指针之间的转换-创新互联
本文路径:http://azwzsj.com/article/ddiihg.html