事件总线之Autofac解耦

事件总线是通过一个中间服务,剥离了常规事件的发布与订阅(消费)强依赖关系的一种技术实现。事件总线的基础知识可参考圣杰的博客【事件总线知多少】

本片博客不再详细概述事件总线基础知识,核心点放置使用Autofac组件实现事件总线与事件事件处理对象的解耦,并在实际业务场景中使用事件总线解决需求。

案例需求

这里还是先面向业务来针对性的探究下事件总线在实际业务场景里的用处有多大,再来讲解后续的Autofac解耦。

在基础数据管理模块里,需要对产品类别删除后也将相应的产品类别标签、产品类别下的产品进行删除。

我们过往的做法是在产品类别删除的业务逻辑后继续编写删除产品类别标签、产品类别对应的产品,类似于下面的代码。

1
2
3
4
5
6
private void EventBusTest1(long productCategoryId)
{                    
     _logger.LogDebug($"删除产品类别{productCategoryId}.");            
     _logger.LogDebug("删除产品类别标签..");            
     _logger.LogDebug("删除产品类别与产品关系..");
}

这种做法本身可以实现我们的实际需求,但是试想如果这个时候我需要再加一个功能,针对删除产品类别后邮件通知管理员。我们发现要实现此功能得继续在之前的代码块中加入通知邮件的相关服务。如下:

  1. private void EventBusTest1(long productCategoryId)
  2. {
  3. _logger.LogDebug($"删除产品类别{productCategoryId}.");
  4. _logger.LogDebug("删除产品类别标签..");
  5. _logger.LogDebug("删除产品类别与产品关系..");
  6. _logger.LogDebug("发送邮件通知管理员..");
  7. }

上面的范例代码是将业务代码采用Logger打印日志模拟出来。如上代码,实际工作中也因为经常性对某一事件进行业务逻辑补充而无限扩张部分代码块,到后来还要进行大面积重构。

ok,如何让我们产品因后续某项事件处理补充而不影响原有的代码块呢?不着急,捋一捋逻辑。

我们先思考下,删除产品类别是不是一个事件,那么它是一定有事件源对象,事件源对象即是我们针对删除产品类别所需业务处理的参数。无它我们什么也做不了,事件源EventData就是这么重要,脑海里想象下某个微信公众号,事件源就是这个公众号发布的文章。那么上面的代码中,入参productCategoryId就是一个事件源(当然我们可以包装成一个对象,补充多一点信息)

那么有了EventData后还是要把业务逻辑从聚合状态分解,不然还是聚合在一起处理和原来的处理方式一样。理想情况下是删除产品类别本身是一个Handler,删除产品类别也是一个Handler,产出产品类别与产品关系及发送邮件通知也是如此。只要触发删除产品类别,则这些Handler也一并会执行。

同一个事件源但凡绑定了多个Handler,则这些Handler会根据事件源进行各自的逻辑处理。

上面对EventData和EventHandler进行了简单介绍,但是我们要怎么触发呢?事件源与处理程序对象如何映射呢?

答案是采取EventBus统一对事件源进行事件触发,事件源于处理程序对象映射在一个字典里。触发事件时EventBus从字典中获取事件源对应的处理对象来分发处理。

哦豁,概念终于唠叨完了,看看使用了事件总线后是如何处理的吧,上码说话。。

先定义事件源接口

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
/// <summary>
/// 定义事件源接口,所有的事件源都要实现该接口
/// </summary>
public interface IEventData
{
    /// <summary>
    /// 事件发生的时间
    /// </summary>
    DateTime EventTime { getset; }
 
    /// <summary>
    /// 触发事件的对象
    /// </summary>
    object EventSource { getset; }
}

再定义事件源基类

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
/// <summary>
/// 事件源基类:描述事件信息,用于参数传递
/// </summary>
public class EventData : IEventData
{
    /// <summary>
    /// 事件发生的时间
    /// </summary>
    public DateTime EventTime { getset; }
 
    /// <summary>
    /// 触发事件的对象
    /// </summary>
    public Object EventSource { getset; }
 
    public EventData()
    {
        EventTime = DateTime.Now;
    }
}

  

定义事件处理接口

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
/// <summary>
/// 定义事件处理器公共接口,所有的事件处理都要实现该接口
/// </summary>
public interface IEventHandler: IDependency 
{
}
 
/// <summary>
/// 泛型事件处理器接口
/// </summary>
/// <typeparam name="TEventData"></typeparam>
public interface IEventHandler<TEventData> : IEventHandler where TEventData : IEventData
{
    /// <summary>
    /// 事件处理器实现该方法来处理事件
    /// </summary>
    /// <param name="eventData"></param>
    void HandleEvent(TEventData eventData);
}

  

定义事件源与事件处理对象存储容器接口

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
/// <summary>
/// 定义事件源与事件处理对象存储容器接口
/// </summary>
public interface IEventStore 
{
    void AddRegister<T, TH>(string keyName) where T : IEventData where TH : IEventHandler;
 
    void AddRegister(Type eventData, string handlerName, Type eventHandler);
 
 
    void RemoveRegister<T, TH>() where T : IEventData where TH : IEventHandler;
 
    void RemoveRegister(Type eventData, Type eventHandler);
 
    bool HasRegisterForEvent<T>() where T : IEventData;
 
    bool HasRegisterForEvent(Type eventData);
 
    IEnumerable<Type> GetHandlersForEvent<T>() where T : IEventData;
 
    IEnumerable<Type> GetHandlersForEvent(Type eventData);
 
    Type GetEventTypeByName(string eventName);
 
    bool IsEmpty { get; }
 
    void Clear();
}

实现IEventStore,这里将事件处理对象与事件源映射信息存储在内存中(无需持久化)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
public class InMemoryEventStore : IEventStore
 {
     /// <summary>
     /// 定义锁对象
     /// </summary>
     private static readonly object LockObj = new object();
 
     private readonly ConcurrentDictionary<ValueTuple<Type, string>, Type> _eventAndHandlerMapping;
 
     public InMemoryEventStore()
     {
         _eventAndHandlerMapping = new ConcurrentDictionary<ValueTuple<Type, string>, Type>();
     }
 
     public void AddRegister<T, TH>(string keyName) where T : IEventData where TH : IEventHandler
     {
         AddRegister(typeof(T), keyName, typeof(TH));
     }
 
     public void AddRegister(Type eventData, string handlerName, Type eventHandler)
     {
         lock (LockObj)
         {
             var mapperKey = new ValueTuple<Type, string>(eventData, handlerName);
             //是否存在事件参数对应的事件处理对象
             if (!HasRegisterForEvent(eventData))
             {
                 _eventAndHandlerMapping.TryAdd(mapperKey, eventHandler);
             }
             else
             {
                 _eventAndHandlerMapping[mapperKey] = eventHandler;
             }
         }
     }
 
     public void RemoveRegister<T, TH>() where T : IEventData where TH : IEventHandler
     {
         var handlerToRemove = FindRegisterToRemove(typeof(T), typeof(TH));
         RemoveRegister(typeof(T), handlerToRemove);
     }
 
 
     public void RemoveRegister(Type eventData, Type eventHandler)
     {
         if (eventHandler != null)
         {
             lock (LockObj)
             {
                 //移除eventHandler
                 var eventHandelerBind = _eventAndHandlerMapping.FirstOrDefault(p => p.Value == eventHandler);
                 if (eventHandelerBind.Value != null)
                 {
                     Type removedHandlers;
                     _eventAndHandlerMapping.TryRemove(eventHandelerBind.Key, out removedHandlers);
                 }
             }
         }
     }
 
     private Type FindRegisterToRemove(Type eventData, Type eventHandler)
     {
         if (!HasRegisterForEvent(eventData))
         {
             return null;
         }
         return _eventAndHandlerMapping.FirstOrDefault(p => p.Value == eventHandler).Value;
     }
 
     public bool HasRegisterForEvent<T>() where T : IEventData
     {
         var mapperDto = _eventAndHandlerMapping.FirstOrDefault(p => p.Key.Item1 == typeof(T));
         return mapperDto.Value != null true false;
     }
 
     public bool HasRegisterForEvent(ValueTuple<Type, string> mapperKey)
     {
         return _eventAndHandlerMapping.ContainsKey(mapperKey);
     }
 
     public bool HasRegisterForEvent(Type eventData)
     {
         return _eventAndHandlerMapping.Count(p => p.Key.Item1 == eventData) > 0 ? true false;
     }
 
     public IEnumerable<Type> GetHandlersForEvent<T>() where T : IEventData
     {
         return GetHandlersForEvent(typeof(T));
     }
 
     public IEnumerable<Type> GetHandlersForEvent(Type eventData) 
     {
         if (HasRegisterForEvent(eventData))
         {
             var items = _eventAndHandlerMapping
                   .Where(p => p.Key.Item1 == eventData)
                   .Select(p => p.Value).ToList();
             return items;
         }
         return new List<Type>();
     }
 
     public Type GetEventTypeByName(string eventName)
     {
         return _eventAndHandlerMapping.Keys.FirstOrDefault(eh => eh.Item2 == eventName).Item1;
     }
 
     public bool IsEmpty => !_eventAndHandlerMapping.Keys.Any();
 
     public void Clear() => _eventAndHandlerMapping.Clear();
       
 }

  

在定义EventHandler,我们这里定义3个EventHandler,分别对应产品类别删除、产品类别标签删除、产品类别与产品关系信息删除

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
public class DeleteProductCategoryEventHandler : IEventHandler<DeleteProductCategoryEventData>
   {
       private readonly ILogger _logger;
 
       public DeleteProductCategoryEventHandler(ILogger<DeleteProductCategoryEventHandler> logger)
       {
           _logger = logger;
       }
       public void HandleEvent(DeleteProductCategoryEventData eventData)
       {
           _logger.LogDebug($"删除产品类别{eventData.ProductCategoryId}..");  
       }
   }
 
 
   
   public class DeleteProductCategoryTagEventHandler : IEventHandler<DeleteProductCategoryEventData>
   {
       private readonly ILogger _logger;
 
       public DeleteProductCategoryTagEventHandler(ILogger<DeleteProductCategoryTagEventHandler> logger)
       {
           _logger = logger;
       }
       public void HandleEvent(DeleteProductCategoryEventData eventData)
       {
           _logger.LogDebug($"删除产品类别{eventData.ProductCategoryId}标签..");       
       }
   }
 
 
   public class DeleteProductCategoryRelEventHandler : IEventHandler<DeleteProductCategoryEventData>
   {
       private readonly ILogger _logger;
 
       public DeleteProductCategoryRelEventHandler(ILogger<DeleteProductCategoryRelEventHandler> logger)
       {
           _logger = logger;
       }
       public void HandleEvent(DeleteProductCategoryEventData eventData)
       {
           _logger.LogDebug($"删除产品类别{eventData.ProductCategoryId}与产品关系..");
       }
   }

  

还有最重要的EventBus,我们使用它来同一触发Handler,EventBus会从EventStore获取到事件源映射的Handler集合,并通过DI容器实例化对象后执行事件。

EventBus

  

当然最重要的是如何使用DI容器注入事件处理程序以及相关依赖服务,这里要注意Autofac对同一个容器只允许Build一次或Update一次。即不允许在程序运行时动态注入服务。且针对同一接口除非使用命名方式注入服务,否则默认一个接口映射一个服务。Autofac这个特性极大程度限制了同一事件源多个Handler的情况,没办法我还是想了个办法,参考AutoMapper的Profile映射方式,定义一个用于映射的类,如下:

通过上面的映射后,在程序初始化时可使用C#反射的特性执行 RegisterEventHandler实现注入同一事件源多个事件处理服务,简直完美~~

下面是Autofac注入的代码:

  

OK,看下最终效果,将触发事件写在Action中

F5跑起来,Debug目录下出来了相应的日志:

内容如下:

总结

事件总线是个好东西,使用它可以将业务分散处理,唯一麻烦点在于需要保证业务处理的原子性问题。以前我们直接一个事务套起来就完事,现在可能要采取类似微服务中的“最终一致性”方案,所以合适的业务采用合适的技术。
时间总线也是领域驱动设计里重要的知识点,掌握它很有必要,微服务中也是采取它配合消息队列实现跨服务处理领域事件。这里使用Autofac也是因为自己的架构本身就是使用它做默认的DI容器,在与事件总线融合中也发现了Autofac的局限性,那就是运行时注册,如果能运行时注册服务的话真的是不要太爽了。

1. 引言

事件总线这个概念对你来说可能很陌生,但提到观察者(发布-订阅)模式,你也许就很熟悉。事件总线是对发布-订阅模式的一种实现。它是一种集中式事件处理机制,允许不同的组件之间进行彼此通信而又不需要相互依赖,达到一种解耦的目的。

我们来看看事件总线的处理流程:

了解了事件总线的基本概念和处理流程,下面我们就来分析下如何去实现事件总线。

2.回归本质

在动手实现事件总线之前,我们还是要追本溯源,探索一下事件的本质和发布订阅模式的实现机制。

2.1.事件的本质

我们先来探讨一下事件的概念。都是读过书的,应该都还记得记叙文的六要素:时间、地点、人物、事件(起因、经过、结果)。

我们拿注册的案例,来解释一下。
用户输入用户名、邮箱、密码后,点击注册,输入无误校验通过后,注册成功并发送邮件给用户,要求用户进行邮箱验证激活。

这里面就涉及了两个主要事件:

  1. 注册事件:起因是用户点击了注册按钮,经过是输入校验,结果是是否注册成功。
  2. 发送邮件事件:起因是用户使用邮箱注册成功需要验证邮箱,经过是邮件发送,结果是邮件是否发送成功。

其实这六要素也适用于我们程序中事件的处理过程。开发过WinForm程序的都知道,我们在做UI设计的时候,从工具箱拖入一个注册按钮(btnRegister),双击它,VS就会自动帮我们生成如下代码:

  1. void btnRegister_Click(object sender, EventArgs e)
  2. {
  3. // 事件的处理
  4. }

其中object sender指代发出事件的对象,这里也就是button对象;EventArgs e 事件参数,可以理解为对事件的描述 ,它们可以统称为事件源。其中的代码逻辑,就是对事件的处理。我们可以统称为事件处理

说了这么多,无非是想透过现象看本质:事件是由事件源和事件处理组成

2.2. 发布订阅模式

定义对象间一种一对多的依赖关系,使得每当一个对象改变状态,则所有依赖于它的对象都会得到通知并被自动更新。 ——发布订阅模式

发布订阅模式主要有两个角色:

  • 发布方(Publisher):也称为被观察者,当状态改变时负责通知所有订阅者。
  • 订阅方(Subscriber):也称为观察者,订阅事件并对接收到的事件进行处理。

发布订阅模式有两种实现方式:

  • 简单的实现方式:由Publisher维护一个订阅者列表,当状态改变时循环遍历列表通知订阅者。
  • 委托的实现方式:由Publisher定义事件委托,Subscriber实现委托。

总的来说,发布订阅模式中有两个关键字,通知和更新。
被观察者状态改变通知观察者做出相应更新。
解决的是当对象改变时需要通知其他对象做出相应改变的问题。

如果画一个图来表示这个流程的画,图形应该是这样的:

3 实现发布订阅模式

相信通过上面的解释,对事件和发布订阅模式有了一个大概的印象。都说理论要与实践相结合,所以我们还是动动手指敲敲代码比较好。
我将以『观察者模式』来钓鱼这个例子为基础,通过重构的方式来完善一个更加通用的发布订阅模式。
先上代码:

  1. /// <summary>
  2. /// 鱼的品类枚举
  3. /// </summary>
  4. public enum FishType
  5. {
  6. 鲫鱼,
  7. 鲤鱼,
  8. 黑鱼,
  9. 青鱼,
  10. 草鱼,
  11. 鲈鱼
  12. }

钓鱼竿的实现:

  1. /// <summary>
  2. /// 鱼竿(被观察者)
  3. /// </summary>
  4. public class FishingRod
  5. {
  6. public delegate void FishingHandler(FishType type); //声明委托
  7. public event FishingHandler FishingEvent; //声明事件
  8. public void ThrowHook(FishingMan man)
  9. {
  10. Console.WriteLine("开始下钩!");
  11. //用随机数模拟鱼咬钩,若随机数为偶数,则为鱼咬钩
  12. if (new Random().Next() % 2 == 0)
  13. {
  14. var type = (FishType) new Random().Next(0, 5);
  15. Console.WriteLine("铃铛:叮叮叮,鱼儿咬钩了");
  16. if (FishingEvent != null)
  17. FishingEvent(type);
  18. }
  19. }
  20. }

垂钓者:

  1. /// <summary>
  2. /// 垂钓者(观察者)
  3. /// </summary>
  4. public class FishingMan
  5. {
  6. public FishingMan(string name)
  7. {
  8. Name = name;
  9. }
  10. public string Name { get; set; }
  11. public int FishCount { get; set; }
  12. /// <summary>
  13. /// 垂钓者自然要有鱼竿啊
  14. /// </summary>
  15. public FishingRod FishingRod { get; set; }
  16. public void Fishing()
  17. {
  18. this.FishingRod.ThrowHook(this);
  19. }
  20. public void Update(FishType type)
  21. {
  22. FishCount++;
  23. Console.WriteLine("{0}:钓到一条[{2}],已经钓到{1}条鱼了!", Name, FishCount, type);
  24. }
  25. }

场景类也很简单:

  1. //1、初始化鱼竿
  2. var fishingRod = new FishingRod();
  3. //2、声明垂钓者
  4. var jeff = new FishingMan("圣杰");
  5. //3.分配鱼竿
  6. jeff.FishingRod = fishingRod;
  7. //4、注册观察者
  8. fishingRod.FishingEvent += jeff.Update;
  9. //5、循环钓鱼
  10. while (jeff.FishCount < 5)
  11. {
  12. jeff.Fishing();
  13. Console.WriteLine("-------------------");
  14. //睡眠5s
  15. Thread.Sleep(5000);
  16. }

代码很简单,相信你一看就明白。但很显然这个代码实现仅适用于当前这个钓鱼场景,假如有其他场景也想使用这个模式,我们还需要重新定义委托,重新定义事件处理,岂不很累。本着”Don't repeat yourself“的原则,我们要对其进行重构。

结合我们对事件本质的探讨,事件是由事件源和事件处理组成。针对我们上面的案例来说,public delegate void FishingHandler(FishType type);这句代码就已经说明了事件源和事件处理。事件源就是FishType type,事件处理自然是注册到FishingHandler上面的委托实例。
问题找到了,很显然是我们的事件源和事件处理不够抽象,所以不能通用,下面咱们就来动手改造。

3.1. 提取事件源

事件源应该至少包含事件发生的时间和触发事件的对象。
我们提取IEventData接口来封装事件源:

  1. /// <summary>
  2. /// 定义事件源接口,所有的事件源都要实现该接口
  3. /// </summary>
  4. public interface IEventData
  5. {
  6. /// <summary>
  7. /// 事件发生的时间
  8. /// </summary>
  9. DateTime EventTime { get; set; }
  10. /// <summary>
  11. /// 触发事件的对象
  12. /// </summary>
  13. object EventSource { get; set; }
  14. }

自然我们应该给一个默认的实现EventData

  1. /// <summary>
  2. /// 事件源:描述事件信息,用于参数传递
  3. /// </summary>
  4. public class EventData : IEventData
  5. {
  6. /// <summary>
  7. /// 事件发生的时间
  8. /// </summary>
  9. public DateTime EventTime { get; set; }
  10. /// <summary>
  11. /// 触发事件的对象
  12. /// </summary>
  13. public Object EventSource { get; set; }
  14. public EventData()
  15. {
  16. EventTime = DateTime.Now;
  17. }
  18. }

针对Demo,扩展事件源如下:

  1. public class FishingEventData : EventData
  2. {
  3. public FishType FishType { get; set; }
  4. public FishingMan FisingMan { get; set; }
  5. }

完成后,我们就可以去把在FishingRod声明的委托参数类型改为FishingEventData类型了,即public delegate void FishingHandler(FishingEventData eventData); //声明委托
然后修改FishingManUpdate方法按委托定义的参数类型修改即可,代码我就不放了,大家自行脑补。

到这一步我们就统一了事件源的定义方式。

3.2.提取事件处理器

事件源统一了,那事件处理也得加以限制。比如如果随意命名事件处理方法名,那在进行事件注册的时候还要去按照委托定义的参数类型去匹配,岂不麻烦。

我们提取一个IEventHandler接口:

  1. /// <summary>
  2. /// 定义事件处理器公共接口,所有的事件处理都要实现该接口
  3. /// </summary>
  4. public interface IEventHandler
  5. {
  6. }

事件处理要与事件源进行绑定,所以我们再来定义一个泛型接口:

  1. /// <summary>
  2. /// 泛型事件处理器接口
  3. /// </summary>
  4. /// <typeparam name="TEventData"></typeparam>
  5. public interface IEventHandler<TEventData> : IEventHandler where TEventData : IEventData
  6. {
  7. /// <summary>
  8. /// 事件处理器实现该方法来处理事件
  9. /// </summary>
  10. /// <param name="eventData"></param>
  11. void HandleEvent(TEventData eventData);
  12. }

你可能会纳闷,为什么先定义了一个空接口?这里就留给自己思考吧。

至此我们就完成了事件处理的抽象。我们再继续去改造我们的Demo。我们让FishingMan实现IEventHandler接口,然后修改场景类中将fishingRod.FishingEvent += jeff.Update;改为fishingRod.FishingEvent += jeff.HandleEvent;即可。代码改动很简单,同样在此略去。

至此你可能觉得我们完成了对Demo的改造。但事实上呢,我们还要弄清一个问题——如果这个FishingMan订阅的有其他的事件,我们该如何处理?
聪颖如你,你立马想到了可以通过事件源来进行区分处理

  1. public class FishingMan : IEventHandler<IEventData>
  2. {
  3. //省略其他代码
  4. public void HandleEvent(IEventData eventData)
  5. {
  6. if (eventData is FishingEventData)
  7. {
  8. //do something
  9. }
  10. if(eventData is XxxEventData)
  11. {
  12. //do something else
  13. }
  14. }
  15. }

至此,这个模式实现到这个地步基本已经可以通用了。

4. 实现事件总线

通用的发布订阅模式不是我们的目的,我们的目的是一个集中式的事件处理机制,且各个模块之间相互不产生依赖。那我们如何做到呢?同样我们还是一步一步的进行分析改造。

4.1.分析问题

思考一下,每次为了实现这个模式,都要完成以下三步:

  1. 事件发布方定义事件委托
  2. 事件订阅方定义事件处理逻辑
  3. 显示的订阅事件

虽然只有三步,但这三步已经很繁琐了。而且事件发布方和事件订阅方还存在着依赖(体现在订阅者要显示的进行事件的注册和注销上)。而且当事件过多时,直接在订阅者中实现IEventHandler接口处理多个事件逻辑显然不太合适,违法单一职责原则。这里就暴露了三个问题:

  1. 如何精简步骤?
  2. 如何解除发布方与订阅方的依赖?
  3. 如何避免在订阅者中同时处理多个事件逻辑?

带着问题思考,我们就会更接近真相。

想要精简步骤,那我们需要寻找共性。共性就是事件的本质,也就是我们针对事件源和事件处理提取出来的两个接口。

想要解除依赖,那就要在发布方和订阅方之间添加一个中介。

想要避免订阅者同时处理过多事件逻辑,那我们就把事件逻辑的处理提取到订阅者外部。

思路有了,下面我们就来实施吧。

4.2.解决问题

本着先易后难的思想,我们下面就来解决以上问题。

4.2.1. 实现IEventHandler

我们先解决上面的第三个问题:如何避免在订阅者中同时处理多个事件逻辑?

自然是针对不同的事件源IEventData实现不同的IEventHandler。改造后的钓鱼事件处理逻辑如下:

  1. /// <summary>
  2. /// 钓鱼事件处理
  3. /// </summary>
  4. public class FishingEventHandler : IEventHandler<FishingEventData>
  5. {
  6. public void HandleEvent(FishingEventData eventData)
  7. {
  8. eventData.FishingMan.FishCount++;
  9. Console.WriteLine("{0}:钓到一条[{2}],已经钓到{1}条鱼了!",
  10. eventData.FishingMan.Name, eventData.FishingMan.FishCount, eventData.FishType);
  11. }
  12. }

这时我们就可以移除在FishingMan中实现的IEventHandler接口了。
然后将事件注册改为fishingRod.FishingEvent += new FishingEventHandler().HandleEvent;即可。

4.2.2. 统一注册事件

上一个问题的解决,有助于我们解决第一个问题:如何精简流程?
为什么呢,因为我们是根据事件源定义相应的事件处理的。也就是我们之前说的可以根据事件源来区分事件。
然后呢?反射,我们可以通过反射来进行事件的统一注册。
FishingRod的构造函数中使用反射,统一注册实现了IEventHandler<FishingEventData>类型的实例方法HandleEvent

  1. public FishingRod()
  2. {
  3. Assembly assembly = Assembly.GetExecutingAssembly();
  4. foreach (var type in assembly.GetTypes())
  5. {
  6. if (typeof(IEventHandler).IsAssignableFrom(type))//判断当前类型是否实现了IEventHandler接口
  7. {
  8. Type handlerInterface = type.GetInterface("IEventHandler`1");//获取该类实现的泛型接口
  9. Type eventDataType = handlerInterface.GetGenericArguments()[0]; // 获取泛型接口指定的参数类型
  10. //如果参数类型是FishingEventData,则说明事件源匹配
  11. if (eventDataType.Equals(typeof(FishingEventData)))
  12. {
  13. //创建实例
  14. var handler = Activator.CreateInstance(type) as IEventHandler<FishingEventData>;
  15. //注册事件
  16. FishingEvent += handler.HandleEvent;
  17. }
  18. }
  19. }
  20. }

这样,我们就可以移出场景类中的显示注册代码fishingRod.FishingEvent += new FishingEventHandler().HandleEvent;

4.2.3. 解除依赖

如何解除依赖呢?其实答案就在本文的两张图上,仔细对比我们可以很直观的看到,Event Bus就相当于一个介于Publisher和Subscriber中间的桥梁。它隔离了Publlisher和Subscriber之间的直接依赖,接管了所有事件的发布和订阅逻辑,并负责事件的中转。

Event Bus终于要粉墨登场了!!!
分析一下,如果EventBus要接管所有事件的发布和订阅,那它则需要有一个容器来记录事件源和事件处理。那又如何触发呢?有了事件源,我们就自然能找到绑定的事件处理逻辑,通过反射触发。代码如下:

  1. /// <summary>
  2. /// 事件总线
  3. /// </summary>
  4. public class EventBus
  5. {
  6. public static EventBus Default => new EventBus();
  7. /// <summary>
  8. /// 定义线程安全集合
  9. /// </summary>
  10. private readonly ConcurrentDictionary<Type, List<Type>> _eventAndHandlerMapping;
  11. public EventBus()
  12. {
  13. _eventAndHandlerMapping = new ConcurrentDictionary<Type, List<Type>>();
  14. MapEventToHandler();
  15. }
  16. /// <summary>
  17. ///通过反射,将事件源与事件处理绑定
  18. /// </summary>
  19. private void MapEventToHandler()
  20. {
  21. Assembly assembly = Assembly.GetEntryAssembly();
  22. foreach (var type in assembly.GetTypes())
  23. {
  24. if (typeof(IEventHandler).IsAssignableFrom(type))//判断当前类型是否实现了IEventHandler接口
  25. {
  26. Type handlerInterface = type.GetInterface("IEventHandler`1");//获取该类实现的泛型接口
  27. if (handlerInterface != null)
  28. {
  29. Type eventDataType = handlerInterface.GetGenericArguments()[0]; // 获取泛型接口指定的参数类型
  30. if (_eventAndHandlerMapping.ContainsKey(eventDataType))
  31. {
  32. List<Type> handlerTypes = _eventAndHandlerMapping[eventDataType];
  33. handlerTypes.Add(type);
  34. _eventAndHandlerMapping[eventDataType] = handlerTypes;
  35. }
  36. else
  37. {
  38. var handlerTypes = new List<Type> { type };
  39. _eventAndHandlerMapping[eventDataType] = handlerTypes;
  40. }
  41. }
  42. }
  43. }
  44. }
  45. /// <summary>
  46. /// 手动绑定事件源与事件处理
  47. /// </summary>
  48. /// <typeparam name="TEventData"></typeparam>
  49. /// <param name="eventHandler"></param>
  50. public void Register<TEventData>(Type eventHandler)
  51. {
  52. List<Type> handlerTypes = _eventAndHandlerMapping[typeof(TEventData)];
  53. if (!handlerTypes.Contains(eventHandler))
  54. {
  55. handlerTypes.Add(eventHandler);
  56. _eventAndHandlerMapping[typeof(TEventData)] = handlerTypes;
  57. }
  58. }
  59. /// <summary>
  60. /// 手动解除事件源与事件处理的绑定
  61. /// </summary>
  62. /// <typeparam name="TEventData"></typeparam>
  63. /// <param name="eventHandler"></param>
  64. public void UnRegister<TEventData>(Type eventHandler)
  65. {
  66. List<Type> handlerTypes = _eventAndHandlerMapping[typeof(TEventData)];
  67. if (handlerTypes.Contains(eventHandler))
  68. {
  69. handlerTypes.Remove(eventHandler);
  70. _eventAndHandlerMapping[typeof(TEventData)] = handlerTypes;
  71. }
  72. }
  73. /// <summary>
  74. /// 根据事件源触发绑定的事件处理
  75. /// </summary>
  76. /// <typeparam name="TEventData"></typeparam>
  77. /// <param name="eventData"></param>
  78. public void Trigger<TEventData>(TEventData eventData) where TEventData : IEventData
  79. {
  80. List<Type> handlers = _eventAndHandlerMapping[eventData.GetType()];
  81. if (handlers != null && handlers.Count > 0)
  82. {
  83. foreach (var handler in handlers)
  84. {
  85. MethodInfo methodInfo = handler.GetMethod("HandleEvent");
  86. if (methodInfo != null)
  87. {
  88. object obj = Activator.CreateInstance(handler);
  89. methodInfo.Invoke(obj, new object[] { eventData });
  90. }
  91. }
  92. }
  93. }
  94. }

事件总线主要定义三个方法,注册、取消注册、事件触发。还有一点就是我们在构造函数中通过反射去进行事件源和事件处理的绑定。
代码注释已经很清楚了,这里就不过多解释了。

下面我们就来修改Demo,修改FishingRod的事件触发:

  1. /// <summary>
  2. /// 下钩
  3. /// </summary>
  4. public void ThrowHook(FishingMan man)
  5. {
  6. Console.WriteLine("开始下钩!");
  7. //用随机数模拟鱼咬钩,若随机数为偶数,则为鱼咬钩
  8. if (new Random().Next() % 2 == 0)
  9. {
  10. var a = new Random(10).Next();
  11. var type = (FishType)new Random().Next(0, 5);
  12. Console.WriteLine("铃铛:叮叮叮,鱼儿咬钩了");
  13. if (FishingEvent != null)
  14. {
  15. var eventData = new FishingEventData() { FishType = type, FishingMan = man };
  16. //FishingEvent(eventData);//不再需要通过事件委托触发
  17. EventBus.Default.Trigger<FishingEventData>(eventData);//直接通过事件总线触发即可
  18. }
  19. }
  20. }

至此,事件总线的雏形已经形成!

5.事件总线的总结

通过上面一步一步的分析和实践,发现事件总线也不是什么高深的概念,只要我们自己善于思考,勤于动手,也能实现自己的事件总线。
根据我们的实现,大概总结出以下几条:

  1. 事件总线维护一个事件源与事件处理的映射字典;
  2. 通过单例模式,确保事件总线的唯一入口;
  3. 利用反射完成事件源与事件处理的初始化绑定;
  4. 提供统一的事件注册、取消注册和触发接口。

最后,以上事件总线的实现只是一个雏形,还有很多潜在的问题。有兴趣的不妨思考完善一下,我也会继续更新完善,尽情期待!


参考资料

ABP EventBus
DDD~领域事件与事件总线
DDD事件总线的实现

Autofac解耦事件总线的更多相关文章

  1. .Net 事件总线之Autofac解耦

    事件总线是通过一个中间服务,剥离了常规事件的发布与订阅(消费)强依赖关系的一种技术实现.事件总线的基础知识可参考圣杰的博客[事件总线知多少] 本片博客不再详细概述事件总线基础知识,核心点放置使用Aut ...

  2. 基于ASP.NET Core 5.0使用RabbitMQ消息队列实现事件总线(EventBus)

    文章阅读请前先参考看一下 https://www.cnblogs.com/hudean/p/13858285.html 安装RabbitMQ消息队列软件与了解C#中如何使用RabbitMQ 和 htt ...

  3. ABP理论学习之事件总线和领域事件

    返回总目录 本篇目录 事件总线 定义事件 触发事件 处理事件 句柄注册 取消注册 在C#中,我们可以在一个类中定义自己的事件,而其他的类可以注册该事件,当某些事情发生时,可以通知到该类.这对于桌面应用 ...

  4. Guava - EventBus(事件总线)

    Guava在guava-libraries中为我们提供了事件总线EventBus库,它是事件发布订阅模式的实现,让我们能在领域驱动设计(DDD)中以事件的弱引用本质对我们的模块和领域边界很好的解耦设计 ...

  5. Android学习系列(43)--使用事件总线框架EventBus和Otto

    事件总线框架 针对事件提供统一订阅,发布以达到组件间通信的解决方案. 原理 观察者模式. EventBus和Otto 先看EventBus的官方定义: Android optimized event ...

  6. Guava: 事件总线EventBus

    EventBus 直译过来就是事件总线,它使用发布订阅模式支持组件之间的通信,不需要显式地注册回调,比观察者模式更灵活,可用于替换Java中传统的事件监听模式,EventBus的作用就是解耦,它不是通 ...

  7. Java事件总线

    在平时写代码的过程中,我们需要实现这样一种功能:当执行某个逻辑时,希望能够进行其他逻辑的处理.最粗暴的方法是直接依赖其他模块,调用该模块的相应函数或者方法.但是,这样做带来一些问题. 模块间相互依赖, ...

  8. EventBus 事件总线 案例

    简介 地址:https://github.com/greenrobot/EventBus EventBus是一个[发布 / 订阅]的事件总线.简单点说,就是两人[约定]好怎么通信,一人发布消息,另外一 ...

  9. 事件总线帧---Otto

    我们如果这样一种业务场景.如今在做一款及时聊天应用,我们在聊天页面进行收发信息.同一时候也要实时更新前一页面的聊天记录,这时我们该怎样去实现?说说我曾经的实现策略.我使用的是广播接收器BroadCas ...

随机推荐

  1. numpy ndarray

    >>> aarray([[1, 2], [3, 4]])>>> a.shape(2, 2)>>> barray([2, 3])>>&g ...

  2. learning at command AT+CGSN

    AT command AT+CGSN [Purpose]        Learning how to get mobile module international Mobile Equipment ...

  3. POJ 1035 Spell checker 字符串 难度:0

    题目 http://poj.org/problem?id=1035 题意 字典匹配,单词表共有1e4个单词,单词长度小于15,需要对最多50个单词进行匹配.在匹配时,如果直接匹配可以找到待匹配串,则直 ...

  4. windows下《Go Web编程》之Go工作空间

    上篇已配置GOPATH工作空间为D:\mygo,之后练习就会在此目录进行... GOPATH目录下有3个子目录: src:存放源代码(.go .c .h .s等 ) pkg:编译后生成的文件(如.a) ...

  5. day30-python阶段性复习四

    九.函数 函数就是完成特定功能的一个语句组,这组语句可以作为一个单位使用,并且给它取一个名字. 降低编程难度 代码重用 可以通过函数名在程序的不同地方多长执行,这通常叫函数调用(.). 预定义函数 可 ...

  6. django搭建博客

    https://andrew-liu.gitbooks.io/django-blog/content/index.html

  7. Java并发编程_wait/notify和CountDownLatch的比较(三)

     1.wait/notify方法 package sync; import java.util.ArrayList; import java.util.List; public class WaitA ...

  8. intelij idea常用设置

    1.genneral设置 2.自动导包 3.设置显示行号和方法分隔符 4.忽略大小写提示代码 比如:输入str会让其提示String 5.去掉单行显示类,让idea多行显示,容易找到类 6.设置字体及 ...

  9. centos6.6安装hadoop-2.5.0(四、hadoop HA安装)

    操作系统:centos6.6 环境:selinux disabled:iptables off:java 1.8.0_131 安装包:hadoop-2.5.0.tar.gz HA模式下的HADOOP完 ...

  10. 线程简述(Thread)

    线程: 进程是一个正在运行的程序,例如电脑上现在在运行的qq,浏览器,电脑管家,这些都是进程 线程就是每一个进程中的一个执行单元,每一个进程至少一个线程,可以有多个线程,例如浏览器上每一个打开的网页都 ...