【发布时间】:2013-09-01 12:00:10
【问题描述】:
我正在考虑在 C++ 应用程序中实现事件的不同方法。有一个建议是通过通知中心实现集中式事件调度。另一种方法是让事件的来源和目标直接通信。但是,我对通知中心的方法有所保留。我将概述我看到的这两种方法(我很可能对它们有一些误解,我以前从未实现过事件处理)。
a) 直接沟通。事件是其源接口的一部分。对事件感兴趣的对象必须以某种方式获取源类的实例并订阅其事件:
struct Source
{
Event</*some_args_here*/> InterestingEventA;
Event</*some_other_args_here*/> InterestingEventB;
};
class Target
{
public:
void Subscribe(Source& s)
{
s.InterestingEventA += CreateDelegate(&MyHandlerFunction, this);
}
private:
void MyHandlerFunction(/*args*/) { /*whatever*/ }
};
(据我了解,boost::signals、Qt 信号/插槽和 .NET 事件都是这样工作的,但我可能是错的。)
b) 通知中心。事件在其源界面中不可见。所有事件都会被发送到某个通知中心,可能是作为单例实现的(任何关于避免这种情况的建议都将不胜感激),因为它们被解雇了。目标对象不必知道有关源的任何信息;他们通过访问通知中心订阅某些事件类型。一旦通知中心收到一个新事件,它就会通知所有对该特定事件感兴趣的订阅者。
class NotificationCenter
{
public:
NotificationCenter& Instance();
void Subscribe(IEvent& event, IEventTarget& target);
void Unsubscribe(IEvent& event, IEventTarget& target);
void FireEvent(IEvent& event);
};
class Source
{
void SomePrivateFunc()
{
// ...
InterestingEventA event(/* some args here*/);
NotificationCenter::Instance().FireEvent(event);
// ...
}
};
class Target : public IEventTarget
{
public:
Target()
{
NotificationCenter::Instance().Subscribe(InterestingEventA(), *this);
}
void OnEvent(IEvent& event) override {/**/}
};
(我从 Poco 取了“通知中心”这个词,据我所知,它实现了这两种方法)。
我可以看到这种方法的一些优点;目标创建订阅会更容易,因为他们不需要访问源。此外,不会有任何生命周期管理问题:与源不同,通知中心总是比目标更长寿,因此它们的目标总是在其析构函数中取消订阅,而不用担心源是否仍然存在(这是我可以直接看到的一个主要问题沟通)。但是,我担心这种方法会导致代码无法维护,因为:
各种各样的事件,可能彼此完全无关,都会进入这个大水槽。
实现通知中心最明显的方式是单例,因此很难跟踪谁以及何时修改了订阅者列表。
事件在任何界面中都不可见,因此根本无法查看特定事件是否属于任何来源。
由于这些缺点,我担心随着应用程序的增长,跟踪对象之间的连接会变得非常困难(例如,我正在想象试图理解为什么某些特定事件不会触发的问题) .
我正在寻求有关“通知中心”方法的优缺点的建议。是否可维护?它适合各种应用吗?也许有办法改进实施?我所描述的两种方法之间的比较,以及任何其他事件处理建议,都是非常受欢迎的。
【问题讨论】:
-
您应该提供一些设计信息,尤其是您的性能限制、连接数和多线程。总的来说,我认为集中式方法更容易管理。
-
@dzada,没有多线程。现在的连接数大约是一百,但我认为如果创建订阅变得更容易,它会迅速增加,可能是 2 或 300。应用是小游戏,所以性能很重要;但是,到目前为止,这些事件的触发频率还不足以引起严重的性能问题。
标签: c++ events observer-pattern