我写了一个类来存储一个函数指针或一个成员函数指针(不能同时存储两者)。当我存储成员函数指针时,我也存储了一个对象指针(接收器)。
问题是:我事先不知道对象的类型和函数签名,所以我使用模板类型名。对于参数,我使用可变参数模板。
我有一个类似的代码:
template <typename... Args>
class A
{
public:
template <typename Object, typename Ret>
A (Object *receiver, Ret (Object::*mem)(Args...)); // store the member function pointer
template <typename Ret>
A (Ret (*function)(Args...)); // store the global function pointer
void operator()(Args... args) const; // to call the function pointer
// ... more public stuff ...
private:
class UndefinedClass;
typedef void (Undefined::*MemFunPtrType)(Args...)
union
{
struct
{
MemFunPtrType memFunPtr; // the stored member function
void *receiverPtr; // the receiver object
void (*call)(void *, MemFunPtrType, Args...);
} member;
void (*funPtr)(Args...); // the stored global function
};
enum class Type {Function, Member} type;
};
由于只需要一个全局函数或成员函数,因此我将所有内容都放在union
.
在构造函数中,我将成员函数mem
转换为 avoid (Undefined::*)(Args...)
并存储它。我从 std::function 的实现中获得了这个技巧。
使用没有捕获的 lambda,我再次转换为原始类型,对象和函数,我称之为:
typedef Ret (Object::*OriginalMemberType)(Args...);
// ...
member.call = [] (void *receiver, MemFunPtrType memFun, Args... args)
{
(static_cast<Object*>(receiver)->*reinterpret_cast<OriginalMemberType>(memFun))(args...);
}
我将此 lambda 存储在call
函数指针中以在内部调用它operator()
。使用 if-else 语句比较类型数据,然后调用正确的指针。
我知道这有点烦人,但它确实有效。我有很多测试,都通过了。但我担心严格的混叠问题。我做了很多指针转换,我不确定这是否属于未定义的行为。
问题:是否允许从Ret (Object::*)(Args...)
to 转换void (UndefinedClass::*)(Args...)
?(Ret,Object和Args是模板参数)注意我从来没有在没有再次转换为原始类型的情况下调用对象。它仅用于存储。
问题::我必须用 编译-fno-strict-aliasing
吗?如果我必须,而且这是一个模板类,我应该-fno-strict-aliasing
在每个项目中使用这个类吗?char
也许我可以使用一个长度数组sizeof(void (UndefinedClass::*)(Args...))
或类似的东西。
笔记:
- 我在 Archlinux 中使用 gcc 4.8.1。我使用 C++11。
- 我没有在这里使用 std::function 只是为了将占位符与 std::bind 一起使用(我不知道所需的占位符的数量)。毕竟这是一个很好的实践和学习。如果您知道如何使用 std::function 执行此操作,非常欢迎回答。
- 实际上,我在 std::vector 中使用这个类来调用相同签名的许多“回调”(信号/插槽类框架)。
太感谢了。如果需要,我会澄清这个混乱的任何方面:)