DDD事件总线的实现
基本思路:
(1) 在事件总线内部维护着一个事件与事件处理程序相映射的字典。
(2) 利用反射,事件总线会将实现了IEventHandler的处理程序与相应事件关联到一起,相当于实现了事件处理程序对事件的订阅。
(3) 当发布事件时,事件总线会从字典中找出相应的事件处理程序,然后利用反射去调用事件处理程序中的方法。
核心类(事件总线类)
using System;
using System.Collections.Generic;
using System.Linq;
using System.Reflection; namespace Framework.EventBus
{
public class EventBus
{ private static EventBus _eventBus = null; private static Dictionary<Type, List<Type>> _eventMapping = new Dictionary<Type, List<Type>>(); // 在这个字典中,key存储的是事件,而value中存储的是事件处理程序 private EventBus() { }
/// <summary>
/// 单例
/// </summary>
/// <returns></returns>
public static EventBus Instance()
{
if (_eventBus == null)
{
_eventBus = new EventBus();
MapEvent2Handler();
}
return _eventBus;
} /// <summary>
/// 发布
/// 这里没有用到队列之类的东西,使用的是直接调用的方式
/// </summary>
/// <param name="eventData"></param>
public void Publish(BaseEvent eventData)
{
// 找出这个事件对应的处理者
Type eventType = eventData.GetType(); if (_eventMapping.ContainsKey(eventType) == true)
{
foreach (Type item in _eventMapping[eventType])
{
MethodInfo mi = item.GetMethod("Handle");
if (mi != null)
{
object o = Activator.CreateInstance(item);
mi.Invoke(o, new object[] { eventData });
}
} }
} /// <summary>
/// 将事件与事件处理程序映射到一起
/// 使用元数据来进行注册
/// </summary>
static void MapEvent2Handler()
{
Assembly assembly = Assembly.GetExecutingAssembly();
Type[] types = assembly.GetTypes(); foreach (Type type in types)
{
Type handlerInterfaceType = type.GetInterface("IEventHandler`1"); // 事件处理者 if (handlerInterfaceType != null) // 若是事件处理者,则以其泛型参数为key,事件处理者的集合为value添加到映射中
{
Type eventType = handlerInterfaceType.GetGenericArguments()[]; // 这里只有一个
// 查找是否存在key
if (_eventMapping.Keys.Contains(eventType))
{
List<Type> handlerTypes = _eventMapping[eventType];
handlerTypes.Add(type);
_eventMapping[eventType] = handlerTypes;
}
else // 存在则添加
{
List<Type> handlerTypes = new List<Type>();
handlerTypes.Add(type);
_eventMapping.Add(eventType, handlerTypes);
}
}
}
} }
}
核心类(事件基类)
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace Framework.EventBus
{
public class BaseEvent
{ /// <summary>
/// 事件发生的时间
/// </summary>
public DateTime EventTime { get; set; } /// <summary>
/// 事件源
/// </summary>
public object EventSource { get; set; } }
}
核心类(事件处理程序接口)
namespace Framework.EventBus
{
public interface IEventHandler<T>
where T : BaseEvent
{
void Handle(T eventData);
}
}
使用方法
实现接口IEventHandler<T>
using System;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace Framework.EventBus
{
/// <summary>
/// 实现了IEventHandler<OrderAddedEvent>接口,就是订阅了OrderAddedEvent事件
/// </summary>
public class OrderAddedEventHandler1 : IEventHandler<OrderAddedEvent>
{
public void Handle(OrderAddedEvent eventData)
{ Console.WriteLine("\r\n");
Console.WriteLine("订单的数据是:" );
Console.WriteLine(" 订单号:" + eventData.Order.OrderId);
Console.WriteLine(" 订单金额:" + eventData.Order.OrderAmount);
Console.WriteLine(" 下单时间:" + eventData.Order.OrderDateTime); }
}
}
注:实现了IEventHandler<OrderAddedEvent>接口,就是订阅了OrderAddedEvent事件
订单类
public class OrderEntity
{ /// <summary>
/// 订单编号
/// </summary>
public string OrderId { get; set; } /// <summary>
/// 下单日期
/// </summary>
public DateTime OrderDateTime { get; set; } /// <summary>
/// 订单金额
/// </summary>
public decimal OrderAmount { get; set; } }
发布事件
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace Framework.EventBus
{
class Program
{
static void Main(string[] args)
{
EventBus bus = EventBus.Instance(); OrderEntity order = new OrderEntity() { OrderId = "", OrderDateTime = DateTime.Now, OrderAmount = };
bus.Publish(new OrderAddedEvent() { EventTime = DateTime.Now, Order = order }); // 发布OrderAddedEvent事件, Console.Read();
} }
}
运行结果

改进
(1)实现基于msmq的事件总线,使得系统能够进行分布式的事件订阅和发布。
下载
参考资料
aspnetboilerplate
https://github.com/aspnetboilerplate/aspnetboilerplate
分享一个分布式消息总线,基于.NET Socket Tcp的发布-订阅框架
http://www.cxyclub.cn/n/53667/
Guava - EventBus(事件总线)
http://greengerong.com/blog/2014/11/27/guava-eventbus/
DDD~领域事件与事件总线
http://www.cnblogs.com/lori/p/3476703.html
事件总线 EventBus的设计
http://www.cnblogs.com/MartinChen999/archive/2011/12/21/2294034.html
DDD事件总线的实现的更多相关文章
- DDD事件总线
DDD事件总线 基本思路: (1) 在事件总线内部维护着一个事件与事件处理程序相映射的字典. (2) 利用反射,事件总线会将实现了IEventHandler的处理程序与相应事件 ...
- 事件总线(Event Bus)知多少
源码路径:Github-EventBus 简书同步链接 1. 引言 事件总线这个概念对你来说可能很陌生,但提到观察者(发布-订阅)模式,你也许就很熟悉.事件总线是对发布-订阅模式的一种实现.它是一种集 ...
- ABP之事件总线(3)
承接上一篇时间总线的学习,在上一篇中我们实现了取消显式注册事件的方式,采用使用反射的方式.这样的好处可以解除Publisher和Scriber的显式依赖,但是问题又来了,因为我们只有Publisher ...
- Autofac解耦事件总线
事件总线之Autofac解耦 事件总线是通过一个中间服务,剥离了常规事件的发布与订阅(消费)强依赖关系的一种技术实现.事件总线的基础知识可参考圣杰的博客[事件总线知多少] 本片博客不再详细概述事件总线 ...
- 事件总线(Event Bus)
事件总线(Event Bus)知多少 源码路径:Github-EventBus简书同步链接 1. 引言 事件总线这个概念对你来说可能很陌生,但提到观察者(发布-订阅)模式,你也许就很熟悉.事件总线是对 ...
- Lind.DDD.Events事件总线~自动化注册
回到目录 让大叔兴奋的自动化注册 对于领域事件之前说过,在程序启动时订阅(注册)一些事件处理程序,然后在程序的具体位置去发布(触发)它,这是传统的pub/sub模式的体现,当然也没有什么问题,为了让它 ...
- DDD~领域事件与事件总线
回到目录 谈谈它 终于有些眉目了,搜刮了很多牛人的资料,英文的,中文的,民国文的,终于小有成就了,同时也做了个DEMO,领域事件这东西好,但需要你明白它之后才会说好,而对于明白领域事件这件事来说,它的 ...
- Guava - EventBus(事件总线)
Guava在guava-libraries中为我们提供了事件总线EventBus库,它是事件发布订阅模式的实现,让我们能在领域驱动设计(DDD)中以事件的弱引用本质对我们的模块和领域边界很好的解耦设计 ...
- EventBus(事件总线)
EventBus(事件总线) Guava在guava-libraries中为我们提供了事件总线EventBus库,它是事件发布订阅模式的实现,让我们能在领域驱动设计(DDD)中以事件的弱引用本质对我们 ...
随机推荐
- div上下左右居中
链接.father { width: 300px; height:150px; position: relative; } .son { position: absolute; top: 0; rig ...
- html如何实现圆角的百度搜索框?
<form action="http://www.baidu.com/baidu" target="_blank"> <input type= ...
- 【JS】Javascript数组操作
1.提取数组中每个对象的某个属性组成新的数组 如数组为: let arry = [ { name: 'zhao', 'age': 18 }, { name: 'qian', 'age': 19 }, ...
- curl常用命令备忘
#####(输出请求头信息) curl -I xxx-Pro:test xxx$ curl -I https://www.baidu.com/ HTTP/1.1 200 OK Accept-Range ...
- webDriver基本运用
import time from selenium import webdriver import unittest import HTMLTestRunner # webDriver基本应用 cla ...
- 可持久化线段树——区间更新hdu4348
和线段树类似,每个结点也要打lazy标记 但是lazy标记和线段树不一样 具体区别在于可持久化后lazy-tag不用往下传递,而是固定在这个区间并不断累加,变成了这个区间固有的性质(有点像分块的标记了 ...
- 学习笔记_J2EE_Spring(一)_入门
3. Spring概述 3.1. Spring是什么 Spring是一个优秀的高可用的JavaEE轻量级开发框架.提供一站式开发解决方案. 3.2. Spring框架出现的背景 在世 ...
- spring security 学习一
1.配置基本的springboot web项目,加入security5依赖,启动项目 浏览器访问,即可出现一个默认的登录页面 2.什么都没有配置 登录页面哪里来的 一般不知从何入手,就看官方文档里是如 ...
- Linux 允许或者禁止ping
Linux默认是允许Ping响应的,系统是否允许Ping由2个因素决定的:A.内核参数,B.防火墙,需要2个因素同时允许才能允许Ping,2个因素有任意一个禁Ping就无法Ping. 具体的配置方法如 ...
- js数据结构与算法--单链表的实现与应用思考
链表是动态的数据结构,它的每个元素由一个存储元素本身的节点和一个指向下一个元素的引用(也称指针或链接)组成. 现实中,有一些链表的例子. 第一个就是寻宝的游戏.你有一条线索,这条线索是指向寻找下一条线 ...