ABP框架 - 领域事件(EventBus)
本节内容:
在C#里,一个类可以定义自己的事件,然后其它类可以注册它,当某些事情发生时,接收到通知。这对于桌面应用或单机的Windows服务非常有用。但是,对于一个Web应用,它就有点问题,因为对象在一个web请求里创建,并且它们生命周期都很短。所以就难于注册一些类事件,同时,直接注册另一个类的事件,也使得类之间更加藕合。
领域事件一般用来解藕业务逻辑和在应用里发生重要领域修改时发出通知。
EventBus
EventBus是一个单例对象,被所有类触发事件或处理事件时共享。为使用事件总线,你先要引用它,有两种方式。
注入 IEventBus
你可以用依赖注入获取一个IEventBus的引用,这儿我们使用属性注入模式:
public class TaskAppService : ApplicationService
{
public IEventBus EventBus { get; set; }
public TaskAppService()
{
EventBus = NullEventBus.Instance;
}
}
在注入事件总线上,属性注入比构造器注入更合适。你的类可以没有事件总线,NullEventBus实现了空对象模式,当你调用它的方法时,方法里什么也不做。
获取默认实例
如果你不能注入它,可以直接使用EventBus.Default。它是全局的事件总线,使用方式如下所示:
EventBus.Default.Trigger(...); //trigger an event
在任何可能的地方都不建议直接使用EventBus.Default,因为它难于单元测试。
定义事件
在触发一个事件前,你首先要定义它,通过一个继承自EventData的类来表现一个事件。假设当一个任务完成后我们想触发一个事件:
public class TaskCompletedEventData : EventData
{
public int TaskId { get; set; }
}
这个类包含处理事件类所需要的属性,EventData类定义了EventSource(事件源,哪个对象触发了事件)和EventTime(何时触发的)属性。
预定义事件
处理完异常
ABP定义了AbpHandledExceptionData,并当ABP自动处理任何异常时,会触发这个事件,这在你想了解更多异常信息时尤其有用(尽管ABP自动记录了所有异常)。你可以注册这个事件,当异常发生时,发出通知。
实体修改
为实体修改提供了泛型的事件:EntityCreationEventData<Tentity>、EntityCreatedEventData<TEntity>、EntityUpdatingEventData<TEntity>、EntityUpdateEventData<TEntity>、EntityDeletingEventData<TEntity>和EntityDeletedEventData<TEntity>,同样也有EntityChangingEventData<TEntity>和EntityChangedEventData<TEntity>,修改可以是插入、更新或删除。
“ing”事件(例如EntityUpdating)在保存修改(SaveChanges)前触发,所以你可以在这些事件里,通过抛出异常,促使工作单元回滚,阻止操作)。“ed”事件(例如EntityUpdated)在保存修改之后被触发,也就没有机会让工作单元回滚了。
实体修改事件定义在Abp.Events.Bus.Entities命名空间里,并在插入、更新或删除实体时,被ABP自动触发。如果你有一个Person实体,你可以注册EntityCreatedEventData<Person>,当一个新的Person创建并插入到数据库后,就可以收到通知。这些事件也支持继承,如果你有一个继承自Person的Student类,并且注册了EntityCreatedEventData<Person>,当一个Person或Student被插入后,你也会收到通知。
触发事件
触发一个事件很简单:
public class TaskAppService : ApplicationService
{
public IEventBus EventBus { get; set; } public TaskAppService()
{
EventBus = NullEventBus.Instance;
} public void CompleteTask(CompleteTaskInput input)
{
//TODO: complete the task on database...
EventBus.Trigger(new TaskCompletedEventData {TaskId = });
}
}
Trigger方法有几个重载:
EventBus.Trigger<TaskCompletedEventData>(new TaskCompletedEventData { TaskId = 42 }); //Explicitly declare generic argument
EventBus.Trigger(this, new TaskCompletedEventData { TaskId = 42 }); //Set 'event source' as 'this'
EventBus.Trigger(typeof(TaskCompletedEventData), this, new TaskCompletedEventData { TaskId = 42 }); //Call non-generic version (first argument is the type of the event class)
触发事件的另一个方法是:使用AggregateRoot类的DomainEvents集合(查看实体文档的相关小节)。
处理事件
为处理一个事件,你应该实现IEventHandler<T>接口,如下所示:
public class ActivityWriter : IEventHandler<TaskCompletedEventData>, ITransientDependency
{
public void HandleEvent(TaskCompletedEventData eventData)
{
WriteActivity("A task is completed by id = " + eventData.TaskId);
}
}
IEventHandler定义了HandleEvent方法,并像上面那样实现它。
EventBus被整合到依赖注入系统里,如我们上面那样实现ITransientDependency,当一个TaskCompleted事件发生后,它创建一个新的ActivityWriter实例,并调用它的HandleEvent方法,然后释放它,更多信息查看依赖注入。
处理基类事件
EventBus支持事件的继承,例如:你可以创建一个TaskEventData和两个子类:TaskCompletedEventData和TaskCreatedEventData:
public class TaskEventData : EventData
{
public Task Task { get; set; }
} public class TaskCreatedEventData : TaskEventData
{
public User CreatorUser { get; set; }
} public class TaskCompletedEventData : TaskEventData
{
public User CompletorUser { get; set; }
}
然后你可以实现IEventhandler<TaskEventData>来处理这两种事件:
public class ActivityWriter : IEventHandler<TaskEventData>, ITransientDependency
{
public void HandleEvent(TaskEventData eventData)
{
if (eventData is TaskCreatedEventData)
{
//...
}
else if (eventData is TaskCompletedEventData)
{
//...
}
}
}
这也就意味着,你可以实现IEventHandler<EventData>来处理应用中的所有事件,你可能不想这样做,但它是可以做到的。
处理程序异常
在处理程序(Handler)抛出一个/一些异常时,Eventbus触发所有Handler事件,如果只有一个处理程序抛出异常,异常会直接被Trigger方法抛出,如果多个处理程序抛出异常,EventBus只为它们抛出一个AggregateException异常。
处理多个事件
在一个处理程序里你可以处理多个事件,此次,你应该为每个事件实现IEventHandler<T>,例如:
public class ActivityWriter :
IEventHandler<TaskCompletedEventData>,
IEventHandler<TaskCreatedEventData>,
ITransientDependency
{
public void HandleEvent(TaskCompletedEventData eventData)
{
//TODO: handle the event...
} public void HandleEvent(TaskCreatedEventData eventData)
{
//TODO: handle the event...
}
}
处理程序注册
为处理事件,我们必须在事件总线里注册处理程序。
自动
ABP找到所有实现IEVentHandler的类并注册到依赖注入(例如:通过实现ITransientDependency,如上面的示例),然后ABP自动把它们注册到事件总线,当一个事件发生,ABP使用依赖注入得到处理程序的引用,并在事件处理后释放该引用。在ABP里,这是使用事件总线的推荐的方式。
手动
可以手动注册事件,但要小心使用。在一个web应用里,事件注册应当中应用启动里完成。在一个Web请求里,注册事件不是一个好的方式,因为注册类请完成后继续注册,并为每个请求重新注册,这可能会引起问题,因为注册类多次被调用。同时要记住,手动注册不使用依赖注入系统。
事件总线的Register方法有几个重载,最简单的是接受一个委托(或lambda):
EventBus.Register<TaskCompletedEventData>(eventData =>
{
WriteActivity("A task is completed by id = " + eventData.TaskId);
});
“任务完成”事件发生后,这个lambda方法就会被调用。第二个是接受一个实现了IEventHantler<T>的对象:
EventBus.Register<TaskCompletedEventData>(new ActivityWriter());
同样是为事件调用ActivityWriter实例。第三个重载接受两个泛型参数:
EventBus.Register<TaskCompletedEventData, ActivityWriter>();
此次,事件总线为每个事件创建一个新的ActivityWriter,如果它是disposable(可释放),并调用ActivityWriter.Dispose方法。
最后,你可以注册一个事件处理程序工作,负责处理程序的创建。一个处理程序工厂有两个方法:GetHandler和ReleaseHandler。例如:
public class ActivityWriterFactory : IEventHandlerFactory
{
public IEventHandler GetHandler()
{
return new ActivityWriter();
} public void ReleaseHandler(IEventHandler handler)
{
//TODO: release/dispose the activity writer instance (handler)
}
}
还有一个特殊的工厂类IocHandlerFactory。它使用依赖注入系统来创建/释放处理程序。ABP在自动注册里也使用这个类,所以,如果你想使用依赖注入系统,直接使用之前定义的自动注册。
反注册
当你向事件总线注册后,想反注册事件,最简单的方式就是释放Register方法返回的值,例如:
//Register to an event...
var registration = EventBus.Register<TaskCompletedEventData>(eventData => WriteActivity("A task is completed by id = " + eventData.TaskId) ); //Unregister from event
registration.Dispose();
当然,其它地方或其它某个时刻,都可能需要反注册,你可以保存注册对象并在需要时释放它。Register方法的所有重载都返回一个可释放的对象给事件。
EventBus也提供了Unregister方法,使用示例:
//Create a handler
var handler = new ActivityWriter(); //Register to the event
EventBus.Register<TaskCompletedEventData>(handler); //Unregister from event
EventBus.Unregister<TaskCompletedEventData>(handler);
它也提供了重载来反注册委托和工厂。反注册处理程序对象必须是注册时的对象。
最后,EventBus提供了一个UnregisterAll<T>()方法,它反注册一个事件的所有处理程序,UnregisterAll()方法反注册所有事件的所有处理程序。
ABP框架 - 领域事件(EventBus)的更多相关文章
- Abp领域事件(EventBus)源码解析
Abp中使用EventBus来解耦领域中的业务逻辑,也是订阅-发布模式的一种实现.简单来说就是,当我触发一个事件,注册了这个事件的处理器就会被找到并执行. 先看看整体代码结构 其中Entities文件 ...
- ABP在领域事件中异步调用方法抛异常
在领域事件中调用UserRegistrationManager.RegisterAsync抛异常 Call UserRegistrationManager.RegisterAsync() throw ...
- ABP框架 - 领域服务
文档目录 本节内容: 简介 例子 创建一个接口 实现服务 使用应用服务 相关论述 为什么不只用应用服务? 如何强制你使用领域服务? 简介 领域服务(或服务)用来执行领域操作和业务规则.Eric Eva ...
- DDD~领域事件中使用分布式事务
回到目录 对于一个聚合来说,它可能会被附加很多事件,这里我们叫它领域事务,因为一个聚会我们可以把它理解成一个领域,一个业务.对于领域事件不清楚的同学可以看看我的这篇文章<DDD~领域事件与事件总 ...
- ABP的事件总线和领域事件(EventBus & Domain Events)
http://www.aspnetboilerplate.com/Pages/Documents/EventBus-Domain-Events EventBus EventBus是个单例,获得Even ...
- ABP(现代ASP.NET样板开发框架)系列之14、ABP领域层——领域事件(Domain events)
点这里进入ABP系列文章总目录 基于DDD的现代ASP.NET开发框架--ABP系列之14.ABP领域层——领域事件(Domain events) ABP是“ASP.NET Boilerplate P ...
- ABP理论学习之事件总线和领域事件
返回总目录 本篇目录 事件总线 定义事件 触发事件 处理事件 句柄注册 取消注册 在C#中,我们可以在一个类中定义自己的事件,而其他的类可以注册该事件,当某些事情发生时,可以通知到该类.这对于桌面应用 ...
- ABP领域层——领域事件(Domain events)
ABP领域层——领域事件(Domain events) 基于DDD的现代ASP.NET开发框架--ABP系列之14.ABP领域层——领域事件(Domain events) ABP是“ASP.NET B ...
- ABP入门系列(19)——使用领域事件
ABP入门系列目录--学习Abp框架之实操演练 源码路径:Github-LearningMpaAbp 1.引言 最近刚学习了下DDD中领域事件的理论知识,总的来说领域事件主要有两个作用,一是解耦,二是 ...
随机推荐
- Jquery 搭配 css 使用,简单有效
前几篇博客中讲了Jquery的基础和点击实际,下面来说一下和css搭配着来怎么做 还是和往常一样,举个例子 好几个方块,然后设置颜色 <!DOCTYPE html PUBLIC "-/ ...
- 来吧,HTML5之基础标签(上)
什么是html5 HTML 5 是下一代的 HTML.HTML5 仍处于完善之中.然而,大部分现代浏览器已经具备了某些 HTML5 支持. 学习过程中标签的理解 <a>标签 定义超链接, ...
- C# 生成验证码图片时消除锯齿
引言 基于生成图片实现了一个手机号转图片的需求. 内容也很简单,直接用手机号生成一个png图片.就是为了背景透明以便其他地方调用. 有无锯齿主要依靠一句代码:g.TextRenderingHint= ...
- 纸箱堆叠 bzoj 2253
纸箱堆叠 (1s 128MB) box [问题描述] P 工厂是一个生产纸箱的工厂.纸箱生产线在人工输入三个参数 n, p, a 之后,即可自动化生产三边边长为 (a mod P, a^2 mod p ...
- 水平可见直线 bzoj 1007
水平可见直线 (1s 128M) lines [问题描述] 在xoy直角坐标平面上有n条直线L1,L2,...Ln,若在y值为正无穷大处往下看,能见到Li的某个子线段,则称Li为可见的,否则Li为被覆 ...
- Angular (SPA) WebPack模块化打包、按需加载解决方案完整实现
文艺小说-?2F,言情小说-?3F,武侠小说-?9F long long ago time-1-1:A 使用工具,long long A ago time-1-2:A 使用分类工具,long long ...
- unity3d导出到IOS程序下 集成unity3dAR功能
转载自: 来自AR学院(www.arvrschool.com),原文地址为:http://www.arvrschool.com/index.php?c=post&a=modify&ti ...
- python性能检测工具整理
python 运行后出现core dump产生core.**文件,可通过gdb来调试 Using GDB with a core dump having found build/python/core ...
- [原创]Macbook Pro Retina 15吋安装Windows 7和Windows 8.1方法
前言 本以为有Bootcamp神器在手,Macbook装Win系统应该是不在话下,没想到着实折腾了一番.期间因为误操作导致OSX也挂掉进不去只得磁盘全部抹掉网络恢复安装.为了让大家少走弯路,提供个人安 ...
- Git时间(第一次写,这个怎么玩啊)
1.安装 Liunx直接打开shell界面,输入:sudo apt-get install git-core ,按下回车之后输入密码即可完成安装: Windows系统在https://git-for- ...