关于函数指针的一些问题小结

发布时间 2023-06-14 18:48:06作者: strive-sun

最近接到一个需求,使用 sdk 提供的消息回调,一般我们是继承 sdk 的消息类,然后 sdk 的消息回调(虚函数)会在有消息的时候调用回调指针,从而触发回调

不过因为 sdk 那边又对该消息类二次封装了并提供了一些接口,所以在研究二次封装的方法时,遇到了一些有意思的问题,故记录下

typedef void(__stdcall* fSDKLogCallback)(void* ctx, int32_t severity,
                                         const char* message,
                                         uint32_t message_size);

  

std::pair<void*, fSDKLogCallback> sdk_log_cb_;

在编译时,如果删除 __stdcall 调用约定后,

typedef void* fSDKLogCallback(void* ctx, int32_t severity,
                                         const char* message,
                                         uint32_t message_size);

程序会在 std::pair 处报错

 error C2207: 'std::pair<_Ty1,_Ty2>::second': a member of a class template cannot acquire a function type

这是因为 std::pair 的模板参数要求是可复制构造的类型,而函数类型是不可复制构造的。因此,如果你直接使用没有 __stdcall 的函数指针类型作为 std::pair 的模板参数之一,会导致编译错误。

通过将函数指针类型添加 __stdcall 关键字,你可以将其定义为可复制构造的类型,并在 std::pair 的模板参数中使用它,从而避免编译错误。

补充:

__stdcall 是一种调用约定,它指定了函数调用时参数的传递方式、栈的清理方式以及函数调用的规则。在某些情况下,特别是在 Windows 平台的一些 API 中,函数指针的类型定义可能要求使用 __stdcall 调用约定。

当你定义一个函数指针类型 fSDKLogCallback 时,如果你的实际使用场景要求使用 __stdcall 调用约定,那么你必须在函数指针类型的定义中显式添加 __stdcall 关键字。