2

所以我正在学习这个事件管理课程。我正在存储指向签名成员函数的指针列表,void (Event*)其中 Event 只是目前存储一些随机数据的结构。

typedef boost::function<void(Event*)> Callback;
typedef vector<Callback> CallbackList;

class EventManager
{
public:
   template<typename T>
   void RegisterEventHandler(const std::string& type, void (T::*handler)(Event*), T* obj)
   {
      mCallbackList[type].push_back(boost::bind(handler, obj, _1));
   }

   void DispatchEvent(const std::string& type, Event* evt)
   {
      for(CallbackList::iterator it = mCallbackList[type].begin(); it != mCallbackList[type].end(); ++it)
      {
         Callback callback = (*it);
         callback(evt);
      }   
   }
private:
   hash_map<std::string, CallbackList> mCallbackList;
};

我想知道,我是否可以派生不同版本的事件,并将指向这些成员函数的指针传递给这个类?目前我正在尝试这个。

class MouseEvent : public Event
{
public:
   int testMouseData1;
   int testMouseData2;
   int testMouseData3;
};

class HelloWorld 
{
public:
   void Display(MouseEvent* evt)
   {
      cout << "Hello, world!" << endl;
   }
};


int main(void)
{
   MouseEvent* evt = new MouseEvent();

   HelloWorld* world = new HelloWorld();
   eventManager->RegisterEventHandler("testType", &HelloWorld::Display, world);

   return 0;
}

这给了我 XCode 中的以下错误。

错误:没有匹配的函数调用“ EventManager::RegisterEventHandler(const char [9], void (HelloWorld::*)(MouseEvent*), HelloWorld*&)

你知道我如何安全地传递一个指针,该指针在其函数签名中期望派生类吗?谢谢。

4

2 回答 2

2

所以我找到了一个似乎对我有用的解决方案,但我不确定这样做是否完全安全。我更改了 RegisterEventHandler 方法以将我发送的所有函数指针转换为相同的类型......

 template<typename T1, typename T2>
   void RegisterEventHandler(const String& type, T1 handler, T2* obj)
   {
      void (T2::*evtHandler)(Event*) = (void (T2::*)(Event*)) (handler);
      mCallbackList[type].push_back(boost::bind(evtHandler, obj, _1));
   }

现在这一切似乎都按我最初的意图工作了。但我对这一切都很陌生,所以我不完全确定这是否安全。有什么想法吗?谢谢

于 2010-02-06T18:37:37.647 回答
1

如果您的原型需要“事件”类型,那么您需要确保void Display(MouseEvent* evt)函数接受“ Event”类型。因此,将其更改为void Display(Event *evt)Then 在调用中您可以将其类型转换回 a MouseEvent,假设调用者传递了一个实际的MouseEvent,引用为 " Event" 。

其次,我相信您的调用方式可能还有其他问题,RegisterEventHandler因为它在模板中,但您没有指定模板类型。

于 2010-02-06T17:40:23.097 回答