这篇文章给大家介绍成员函数指针的结构以及怎么与普通函数指针之间的转换,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。
通过内存拷贝(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);
}
但不能保证转换的有效性。
关于成员函数指针的结构以及怎么与普通函数指针之间的转换就分享到这里了,希望以上内容可以对大家有一定的帮助,可以学到更多知识。如果觉得文章不错,可以把它分享出去让更多的人看到。
亿速云「云服务器」,即开即用、新一代英特尔至强铂金CPU、三副本存储NVMe SSD云盘,价格低至29元/月。点击查看>>
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。
原文链接:https://www.xuebuyuan.com/3256667.html