一 背景
在上面的一篇文章中我們介紹了一個完整地基於消息發布和訂閱的模型,這篇文章我將介紹一種簡單的基於消息的發布和訂閱模型,在這個模型中我們將通過構建一個Publisher類來完成對特定的事件和事件訂閱進行封裝,這個是一個更加輕量級別的方式,使用這個的主要目的是降低類之間彼此的耦合程度,從而方便代碼的擴展和訪問,最終使代碼結構更加合理。
我們首先來看看具體的Publisher類的構成,后面我們將會對這個類做一個詳細的講解和分析。
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace Pangea.Common.Utility
{
/// <summary>
/// Publisher/Subscriber pattern
/// This infrastructure works as an instance.
/// </summary>
public sealed class Publisher<EventType> where EventType : IComparable
{
private static readonly object _locker = new object();
private static readonly Dictionary<EventType, List<Action<object, object, object, object>>> _eventTypeList = new Dictionary<EventType, List<Action<object, object, object, object>>>();
public static void Attach(EventType key, Action<object, object, object, object> act)
{
lock (_locker)
{
if (!_eventTypeList.ContainsKey(key))
_eventTypeList.Add(key, new List<Action<object, object, object, object>>());
if (!_eventTypeList[key].Contains(act))
_eventTypeList[key].Add(act);
}
}
public static void Detach(EventType key)
{
lock (_locker)
{
if (_eventTypeList.ContainsKey(key))
{
_eventTypeList[key].Clear();
}
}
}
public static void Detach(EventType key, Action<object, object, object, object> act)
{
lock (_locker)
{
if (_eventTypeList.ContainsKey(key) && _eventTypeList[key].Contains(act))
{
_eventTypeList[key].Remove(act);
}
}
}
public static void Notify(EventType key, object p1, object p2, object p3, object p4)
{
lock (_locker)
{
if (_eventTypeList.ContainsKey(key))
{
foreach (var act in _eventTypeList[key])
{
act.Invoke(p1, p2, p3, p4);
}
}
}
}
public static void Notify(EventType key, object p1, object p2, object p3)
{
Notify(key, p1, p2, p3, null);
}
public static void Notify(EventType key, object p1, object p2)
{
Notify(key, p1, p2, null, null);
}
public static void Notify(EventType key, object p1)
{
Notify(key, p1, null, null, null);
}
public static void Notify(EventType key)
{
Notify(key, null, null, null, null);
}
}
}
二 完成分析
在上面的Publisher類中,我們使用了一個EventType類型的泛型參數,這個參數由我們自己來進行定義,一般定義為枚舉類型用於標識我們唯一的事件類型,定義好了泛型參數后我們定義了一個_eventTypeList用於定義具體的EventType和對應的處理方法並且將這個放到一個靜態的Dictionary中,當然我們在代碼中一個事件可以對應多個處理的Action,但是在一般情況下一個事件只有一個對應的處理函數,這里也不排除一個事件對應多個事件的處理函數,另外這里我們默認定義了四個object類型的參數,用於向處理函數進行參數的傳遞,另外我們的方法都是靜態的方法並且在訂閱的過程中加了鎖從而保證了多線程的環境中保證線程的安全性。
三 使用過程
在具體的使用過程中主要包括三個重要的步驟:1 定義事件類型。2 消息接收方訂閱事件處理(一般在構造函數中調用Attach方法創建時間的唯一訂閱)。3 事件發布方發布事件(事件的發送方觸發時間並傳遞參數,調用Notify方式觸發事件),完成以上三個步驟就完成了整個基於消息的發布訂閱的簡單處理,這個Publisher類在處理一般的事件方法中都能夠順利實現,最重要的是事件本身簡單易於理解而且能最大程度上降低模塊間的耦合,所以在使用的時候還是非常方便的。
