原文链接:http://www.titilima.cn/show-537-1.html
今天的话题比较有意思,如何获取一个非 static 成员函数的指针?考虑以下代码:
- classA
- {
- public:
- voidfoo(void);
- };
- voidA::foo(void)
- {
- puts("Hello,World!");
- }
也就是说,如何获取 A::foo 的指针?
那位说了:这有何难?一个 typedef 全搞定!
- typedefvoid(A::*FooPtr)(void);
- FooPtrfunc=&A::foo;
且慢,我还没说完呢。我要把这个指针用于 thunk 技术,所以我希望得到一个 void* 类型的指针。这样一来,如果再使用一个类型转换的话,那么就会得到一个 C2440 的编译错误,因为 C++ 是强类型的,不允许我们胡作非为。
于是我们只能搞些歪门邪道了,且看:
方案一 汇编
- void*func=NULL;
- __asm
- {
- pusheax
- leaeax,A::foo
- movfunc,eax
- popeax
- }
优点:效率高,无废话。
缺点:可移植性差。
方案二 stdio
- void*func=NULL;
- charaddr[9];
- sprintf(addr,"%p",&A::foo);
- sscanf(addr,"%p",&func);
优点:完全可移植。
缺点:效率低,而且有些无厘头。
另外需要指出的是,对于 VC 的 Debug 配置而言,以上的两种方式获取的都不是 A::foo 的真实地址,而是 ILT 的地址。因此如果我们需要的是真实地址的话,还需要另外解析 ILT 表项的代码。
本文探讨了如何获取C++中非静态成员函数的指针,并提供了两种方法:通过汇编指令直接获取指针和利用stdio库函数进行转换。同时分析了这两种方法的优缺点。

1794

被折叠的 条评论
为什么被折叠?



