我想创建一个非常简单的事件总线,允许任何客户端订阅特定类型的事件,当任何发布者使用EventBus.PushEvent()
方法在总线上推送事件时,只有订阅该特定事件的客户端事件类型将获得该事件。
我正在使用C#和.NET 2.0。
答案 0 :(得分:16)
Tiny Messenger是一个不错的选择,我已经在现场项目中使用它已经有2.5年了。来自Wiki的一些代码示例(链接如下):
<强>出版强>
messageHub.Publish(new MyMessage());
<强>订阅强>
messageHub.Subscribe<MyMessage>((m) => { MessageBox.Show("Message Received!"); });
messageHub.Subscribe<MyMessageAgain>((m) => { MessageBox.Show("Message Received!"); }, (m) => m.Content == "Testing");
GitHub上的代码:https://github.com/grumpydev/TinyMessenger
Wiki似乎仍然在BitBucket上:https://bitbucket.org/grumpydev/tinyioc/wiki/TinyMessenger
它还有一个Nuget包
Install-Package TinyMessenger
答案 1 :(得分:5)
另一个,受到EventBus for android的启发,但更简单:
public class EventBus
{
public static EventBus Instance { get { return instance ?? (instance = new EventBus()); } }
public void Register(object listener)
{
if (!listeners.Any(l => l.Listener == listener))
listeners.Add(new EventListenerWrapper(listener));
}
public void Unregister(object listener)
{
listeners.RemoveAll(l => l.Listener == listener);
}
public void PostEvent(object e)
{
listeners.Where(l => l.EventType == e.GetType()).ToList().ForEach(l => l.PostEvent(e));
}
private static EventBus instance;
private EventBus() { }
private List<EventListenerWrapper> listeners = new List<EventListenerWrapper>();
private class EventListenerWrapper
{
public object Listener { get; private set; }
public Type EventType { get; private set; }
private MethodBase method;
public EventListenerWrapper(object listener)
{
Listener = listener;
Type type = listener.GetType();
method = type.GetMethod("OnEvent");
if (method == null)
throw new ArgumentException("Class " + type.Name + " does not containt method OnEvent");
ParameterInfo[] parameters = method.GetParameters();
if (parameters.Length != 1)
throw new ArgumentException("Method OnEvent of class " + type.Name + " have invalid number of parameters (should be one)");
EventType = parameters[0].ParameterType;
}
public void PostEvent(object e)
{
method.Invoke(Listener, new[] { e });
}
}
}
用例:
public class OnProgressChangedEvent
{
public int Progress { get; private set; }
public OnProgressChangedEvent(int progress)
{
Progress = progress;
}
}
public class SomeForm : Form
{
// ...
protected override void OnLoad(EventArgs e)
{
base.OnLoad(e);
EventBus.Instance.Register(this);
}
public void OnEvent(OnProgressChangedEvent e)
{
progressBar.Value = e.Progress;
}
protected override void OnClosed(EventArgs e)
{
base.OnClosed(e);
EventBus.Instance.Unregister(this);
}
}
public class SomeWorkerSomewhere
{
void OnDoWork()
{
// ...
EventBus.Instance.PostEvent(new OnProgressChangedEvent(progress));
// ...
}
}
答案 2 :(得分:2)
我找到了 Generic Message Bus 。这是一个简单的课程。
答案 3 :(得分:2)
您还可以查看Unity扩展程序: http://msdn.microsoft.com/en-us/library/cc440958.aspx
[Publishes("TimerTick")]
public event EventHandler Expired;
private void OnTick(Object sender, EventArgs e)
{
timer.Stop();
OnExpired(this);
}
[SubscribesTo("TimerTick")]
public void OnTimerExpired(Object sender, EventArgs e)
{
EventHandler handlers = ChangeLight;
if(handlers != null)
{
handlers(this, EventArgs.Empty);
}
currentLight = ( currentLight + 1 ) % 3;
timer.Duration = lightTimes[currentLight];
timer.Start();
}
有更好的吗?
答案 4 :(得分:1)
Composite Application Block包含可能对您有用的事件代理。
答案 5 :(得分:1)
另一个很好的实现可以在:
找到http://code.google.com/p/fracture/source/browse/trunk/Squared/Util/EventBus.cs
用例可通过以下方式访问: /trunk/Squared/Util/UtilTests/Tests/EventTests.cs
此实现不需要外部库。
改进可能是订阅类型而不是字符串。
答案 6 :(得分:0)
你应该看一下Hibernating Rhinos第3集,Ayende的银幕系列节目 - “实施活动经纪人”。
它展示了如何使用Windsor实现一个非常简单的事件代理来连接。源代码也包括在内。
建议的事件代理解决方案非常简单,但是不需要花费太多时间来扩充解决方案以允许参数与事件一起传递。
答案 7 :(得分:0)
我创造了这个:
https://github.com/RemiBou/RemiDDD/tree/master/RemiDDD.Framework/Cqrs
ninject有一个依赖性。 你有一个MessageProcessor。如果要查看事件,如果要处理命令工具“ICommandHandleer”,请实现“IObserver”