首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >问答首页 >在成员函数中获取成员函数的地址

在成员函数中获取成员函数的地址
EN

Stack Overflow用户
提问于 2014-08-29 08:46:17
回答 2查看 1K关注 0票数 2

我想在同一个成员函数中获得成员函数的地址。这个指针已经被获取,所以我知道从哪个对象调用我。当然,编译器会抱怨我试图使用指向这个函数的指针,而不是调用它。我明白了在对象之外使用这个规则的意义。有没有办法获得“我自己的”地址(函数地址时,它被调用)地址?

这仅用于调试目的,应在宏中使用。有办法这么做就太好了。

旨在澄清以下事项的守则:

代码语言:javascript
运行
复制
class A
{
    void do_stuff()
    {
        printf("0x%X\n", (void*)(this->do_stuff));
    }
};

这是在嵌入式系统上,我不明白为什么绝对没有办法获得do_stuff被保存在闪存的地址。

编辑我试着获得“成员函数指针”,如下所示:

代码语言:javascript
运行
复制
printf("CAPP::TASK 0x%08X\n", &CApp::Start);

它给了我0x08013E85,我在我的链接地图上查找,这正是我正在寻找的功能。这是mdk-arm使用Keil Vision5作为编译器.这是正确的吗?

编辑2:所有类型的类型转换给我无效的类型转换。虽然Printf的输出是正确的:Printf使用各种宏。让我们重建这个:

代码语言:javascript
运行
复制
uint32_t getAdr(uint32_t n, ...)
{
    uint32_t adr=0;
    va_list vl;
    va_start(vl,n);
    adr=va_arg(vl,uint32_t);
    va_end(vl);
    return adr;
}

#define GetAdr(x) getAdr(0,&x);

void CApp::Start()
{  
    uint32_t a = GetAdr(CApp::Start);
    printf("Adr: 0x%08X\n", a);
    ...
}

很恶心的黑客,但成功了。但为什么这样做呢?遗憾的是,我没有__va_arg(.)的实现。如何将__va_arg类型的值转换为给定类型?有什么方法可以利用这个功能吗?

EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2014-08-29 11:43:07

指针到成员函数通常是普通指针(如void* )大小的两倍,因此在标准C++中不可能进行直接转换。

https://gcc.gnu.org/onlinedocs/gcc/Bound-member-functions.html文档中包含一个GCC扩展,它允许您结合this&CApp::Start获得一个普通(非成员)函数指针(该页面还解释了为什么指向成员函数的指针除了函数地址之外还有其他信息)。

这个扩展可以这样使用:

代码语言:javascript
运行
复制
typedef void (*pf_type)(CApp*);
pf_type pf = (pf_type)&CApp::Start;

我猜您使用va_arg的方法是因为您的编译器将指针到成员的函数&CApp::Start表示为一个双单词对象,函数地址在其中一个单词中,您通过varargs hack读取函数地址,因为两个单词被推到堆栈上,然后您只读取其中一个单词。您可能可以通过联合获得类似的结果:

代码语言:javascript
运行
复制
union dirty_hack
{
  void (CApp::*pmf)();
  uint32_t words[2];
};
dirty_hack x;
x.pmf = &CApp::Start;
printf("Adr: 0x%08X\n", x.words[0]);   // might need to be x.words[1] instead

您可以使用模板来概括这一点:

代码语言:javascript
运行
复制
template<typename F>
uint32_t
getaddr(F f)
{
  union dirty_hack
  {
    F pmf;
    uint32_t words[2];
  };
  dirty_hack x;
  x.pmf = f;
  return x.words[0];   // might need to be x.words[1] instead
}

printf("Adr: 0x%08X\n", getaddr(&CApp::Start));

注:对于指针不是32位的系统,最好使用uintptr_t而不是int32_t来移植,尽管这种黑客本质上是不可移植的(上面的联合技巧依赖于类型双关,这是未定义的行为,但有一些编译器支持)。

票数 3
EN

Stack Overflow用户

发布于 2014-08-29 09:07:53

没有“这个对象的成员函数”这样的东西。成员函数实际上是由同一个类的所有对象共享的--它只是代码,而不是数据。this通常作为额外的隐藏参数传递给这些函数。

实际上,编译器转换如下:

代码语言:javascript
运行
复制
struct Obj
{
  int a;

  int foo(int x) const { return x * a; }
};

int bar(const Obj &o)
{ return o.foo(42); }

变成这样的东西:

代码语言:javascript
运行
复制
struct Obj
{
  int a;

  static int foo(const Obj *this, int x) { return this->x * a; }
};

int bar(const Obj &o)
{ return Obj::foo(&o, 42); }

正如你所看到的,没有"&(o.foo)“这样的东西。obj的所有实例都具有相同的foo

此外,C++不允许您访问这个“静态foo",这正是因为它是一个实现细节,而且编译器可以根据他们的选择(或者根据平台的要求)进行不同的操作。你能得到的最好的是指向函数的指针,&Obj::foo。当然,这不是真正的地址,它更像是类Obj中的偏移量。您需要明确地列出它的名称--如果不使用它的名称,就无法引用“封装函数”。

在不给函数命名的情况下可以得到的最好结果是预定义变量__func__ (在C++11中引入),它包含特定于编译器的函数名称形式。

票数 2
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/25564749

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档