Ответ 1
Я нашел Общая шина сообщений. Это один простой класс.
Я хочу создать очень простую шину событий, которая позволит любому клиенту подписаться на конкретный тип события и когда любой издатель нажимает событие на шине, используя метод EventBus.PushEvent()
, только клиенты, которые подписались на этот тип события получит событие.
Я использую С# и .NET 2.0.
Я нашел Общая шина сообщений. Это один простой класс.
Tiny Messenger - хороший выбор, я использую его в живом проекте уже 2,5 года. Некоторые примеры кода из вики (ссылка ниже):
Издательство
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
Вики здесь: https://github.com/grumpydev/TinyMessenger/wiki
Он также имеет пакет Nuget
Install-Package TinyMessenger
Другой, вдохновленный EventBus для 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));
// ...
}
}
Вы также можете проверить расширения 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();
}
Есть ли лучшие?
Composite Application Block включает в себя брокер событий, который может вам пригодиться.
Еще одна хорошая реализация может быть найдена по адресу:
http://code.google.com/p/fracture/source/browse/trunk/Squared/Util/EventBus.cs
Варианты использования доступны по адресу: /trunk/Squared/Util/UtilTests/Tests/EventTests.cs
Эта реализация не нуждается в внешней библиотеке.
Улучшение может заключаться в возможности подписки на тип, а не на строку.
Вы должны проверить эпизод 3 в "Спящий носорог" , серия экранов "Айенде" - "Реализация брокера событий".
Это показывает, как вы можете реализовать очень простой брокер событий, используя Windsor для подключения к сети. Исходный код также включен.
Предлагаемое решение для брокера событий очень простое, но для увеличения решения, позволяющего передавать аргументы вместе с событиями, не потребуется слишком много часов.
Я создал это:
https://github.com/RemiBou/RemiDDD/tree/master/RemiDDD.Framework/Cqrs
Существует зависимость от ninject. У вас есть MessageProcessor. Если вы хотите наблюдать за событием, выполните "IObserver", если вы хотите обработать командную реализацию "ICommandHandleer"