【发布时间】:2015-07-07 08:05:42
【问题描述】:
我正在创建一个轻量级的跨平台插件框架,它在应用程序和插件之间使用 C 接口(通常,但并非总是用 C++ 编写)。
要帮助 C++ 应用程序和插件编写者,我面临的一个挑战是找到一种简单的方法来跨 C 接口公开 C++ 对象功能。我目前的解决方案感觉很简单,并使用模板来“构建”包装底层 C++ 成员函数的 C 签名函数,基于 this great stackoverflow question and answer
template <typename Tc, typename F, F>
struct MemberFuncWrapper;
template <typename Tc, // C interface structure tag
typename T, // C++ class, derived from Tc
typename R, // C++ member function return type
typename ...Args, // C++ member function argument types
R (T::*f)(Args...) const> // C++ member function
struct MemberFuncWrapper<Tc, R (T::*)(Args...) const, f> {
static R call(const Tc * tc, Args... args) {
const T * t = static_cast<const T *>(tc);
return ((*t).*f)(args...);
}
};
此模板的实例化在 linux (gcc) 和 mac (clang) 下编译和运行良好,但在 Visual Studio 2013 中编译失败:
error C2440: 'specialization' : cannot convert from 'overloaded-function' to 'void (__cdecl Greeter::* )(void) const'
error C2973: 'MemberFuncWrapper<Tc,R(__cdecl T::* )(Args...) const,f>' : invalid template argument 'overloaded-function'
下面的独立示例代码显示了 Visual Studio 失败的行(在Greeter 类定义中)。我希望有人可以:
- 提供一种可以安抚 Visual Studio 的解决方法,或者
- 提出实现类似目标的替代策略
这里的独立代码演示了在相当冗长的Hello world 应用程序中使用 C++ 类实现 C 接口的上下文中使用的模板代码:
#include <iostream>
#include <utility>
//
// C interface and function(s) typically defined elsewhere
//
#ifdef __cplusplus
extern "C" {
#endif
// The C interface implemented by a 'greeter'
struct greeter_c {
void(*greet_cb)(const struct greeter_c * greeter,
const char * recipient);
};
// Some C function that makes use of a greeter
void broadcast(const struct greeter_c * greeter) {
greeter->greet_cb(greeter, "world");
}
#ifdef __cplusplus
} // extern "C"
#endif
//
// Template magic that envelopes a C++ member
// function call in a C-signature function
//
template <typename Tc, typename F, F>
struct MemberFuncWrapper;
template <typename Tc, // C interface structure tag
typename T, // C++ class, derived from Tc
typename R, // C++ member function return type
typename ...Args, // C++ member function argument types
R (T::*f)(Args...) const> // C++ member function
struct MemberFuncWrapper<Tc, R (T::*)(Args...) const, f> {
static R call(const Tc * tc, Args... args) {
// Cast C structure to C++ object
const T * t = static_cast<const T *>(tc);
// Details such as catching/handling exceptions omitted.
// Call C++ member function
return ((*t).*f)(args...);
}
};
// Repeat of the above for non-const member functions omitted
//
// A C++ class that implements the C 'greeter' interface
//
class Greeter : public greeter_c {
public:
// Constructor
Greeter(const char * greeting) : m_greeting(greeting) {
// Set up C interface callback by wrapping member function
// !! The following line causes the Visual Studio compilation error !!
greet_cb = MemberFuncWrapper<greeter_c,
void (Greeter::*)(const char *) const,
&Greeter::greet>::call;
}
// C++ member function that 'does' the greeting
void greet(const char * recipient) const {
std::cout << m_greeting << " " << recipient << std::endl;
}
private:
const char * m_greeting;
};
// An application that greets using a Greeter's C interface
int main(int argc, char * argv[]) {
// Create C++ object that implements C interface
Greeter a("Hello");
// Greet using Greeter's C interface
broadcast(&a);
return 0;
}
技术细节:
- Linux 开发:Centos 7、g++ (GCC) 4.8.3 20140911
- Mac 开发:Apple LLVM 版本 6.1.0 (clang-602.0.49)
- Windows 开发:Visual Studio Express 2013 Update 5 CTP
【问题讨论】:
-
您能否显示导致该错误的具体代码?
-
哦,您不能只使用
std::function并免费获得所有这些东西有什么特别的原因吗?那是在 C++11 中出现的,所以希望 VS 2013 有支持? -
@Useless:我已经编辑了问题,以提请注意示例应用程序中导致 Visual Studio 失败的代码行。
-
@Useless:如何使用 std::function 来实现我通过接受成员函数作为参数的模板实现的目标?我的目标是简洁地为应用程序或插件编写者创建的任何 C++ 对象建立一个 C 接口,尽可能少地重复样板代码。
-
有趣。会想办法解决的。工作...
标签: c++ c plugins visual-studio-2013 interface