EvnetBus

领域事件(EvnetBus)

文档目录

本节内容:

在C#里,一个类可以定义自己的事件,然后其它类可以注册它,当某些事情发生时,接收到通知。这对于桌面应用或单机的Windows服务非常有用。但是,对于一个Web应用,它就有点问题,因为对象在一个web请求里创建,并且它们生命周期都很短。所以就难于注册一些类事件,同时,直接注册另一个类的事件,也使得类之间更加藕合。

领域事件一般用来解藕业务逻辑和在应用里发生重要领域修改时发出通知。

EventBus

EventBus是一个单例对象,被所有类触发事件或处理事件时共享。为使用事件总线,你先要引用它,有两种方式。

注入 IEventBus

你可以用依赖注入获取一个IEventBus的引用,这儿我们使用属性注入模式:

public class TaskAppService : ApplicationService
{
    ; }

    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 TaskAppService()
    {
        EventBus = NullEventBus.Instance;
    }

    public void CompleteTask(CompleteTaskInput input)
    {
        //TODO: complete the task on database...
        EventBus.Trigger(new TaskCompletedEventData {TaskId = 42});
    }
}

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,
    IEventHandler, 
    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()方法反注册所有事件的所有处理程序。

分类: ASP.NET Boilerplate

标签: ABP ASP.NET Boilerplate 框架 文档 领域事件 EventBus Domain

时间: 2024-10-01 03:31:19

EvnetBus的相关文章

&lt;&lt;ABP框架&gt;&gt; 领域事件(EvnetBus)

文档目录 本节内容: EventBus 注入 IEventBus 获取默认实例 定义事件 预定义事件 处理完异常 实体修改 触发事件 处理事件 处理基类事件 处理程序异常 处理多个事件 处理程序注册 自动 手动 反注册 在C#里,一个类可以定义自己的事件,然后其它类可以注册它,当某些事情发生时,接收到通知.这对于桌面应用或单机的Windows服务非常有用.但是,对于一个Web应用,它就有点问题,因为对象在一个web请求里创建,并且它们生命周期都很短.所以就难于注册一些类事件,同时,直接注册另一个

EventBus的使用和原理剖析

尊重原创:http://blog.csdn.net/yuanzeyao/article/details/38174537 代码下载:http://download.csdn.net/detail/yuanzeyao2008/7684041 在编程过程中,当我们想通知其他组件某些事情发生时,我们通常使用观察者模式,正式因为观察者模式非常常见,所以在jdk1.5中已经帮助我们实现了观察者模式,我们只需要简单的继承一些类就可以快速使用观察者模式,在Android中也有一个类似功能的开源库EventBu

Android之EventBus使用详解

一.概述 当Android项目越来越庞大的时候,应用的各个部件之间的通信变得越来越复杂,例如:当某一条件发生时,应用中有几个部件对这个消息感兴趣,那么我们通常采用的就是观察者模式,使用观察者模式有一个弊病就是部件之间的耦合度太高,在这里我将会详细介绍Android中的解耦组建EventBus的使用.EventBus是一款针对Android优化的发布/订阅事件总线.主要功能是替代Intent,Handler,BroadCast在Fragment,Activity,Service,线程之间传递消息.

Android EventBus框架(二)之源码简单解析

上一篇,我们基本知道了EventBus的使用步骤,接下来先简单研究一下其中的源码.在分析源码之前,我们先回顾一下Java反射的知识点: JAVA反射机制 基本定义: JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制. Sun为提供的关于反射机制中的类: java.lang.Class; java.lang.reflect.Construct

EventBus框架源码分析

开源项目 上周又手动撸了一遍EventBus实现,同时上传EventBus的中文注释源码到Github上,欢迎大家fork&star. EventBusAnalysis EventBus 基础概念 EventBus是一个Android事件发布/订阅框架,通过解耦发布者和订阅者简化Android事件传递.事件传递既可以用于Android四大组件间的通讯,也可以用于用户异步线程和主线程间通讯等. 传统的事件传递方法包括:Handler,BroadCastReceiver,interface回调,相比

Fragment使用具体解释

fragment是Google在3.0版本号中推出的新功能,如今已经增加到V4包中,假设要使用V4兼容包中的Fragment须要将Activity换成FragmentActivity,调用的getSupportFragmentManager获取FragmentManager而不是getFragmentManager. Fragment是Google大力推荐使用的一个功能.它和Activity功能事实上差点儿相同,只是比Activity更加的灵活和轻巧,Fragment寄存在Activity上面,

EventBus使用详情、源码分析和注意事项

基本介绍 EventBus主要用于事件的订阅和发布,主要功能是替代Intent,Handler,BroadCast在Fragment,Activity,Service,线程之间传递消息.官方文档介绍了EventBus的很多优点,归纳一下就是三个优点:小,快和方便.以下会围绕初步使用.使用进阶.源码分析和注意事项来讲解,如果需要快速上手,只要看完初步使用和注意事项就可以了. 初步使用 EventBus的使用非常简单,主要使用到以下三个方法: //注册EventBus EventBus.getDef

EventBus框架库代码走读

PS一句:最终还是选择CSDN来整理发表这几年的知识点,该文章平行迁移到CSDN.因为CSDN也支持MarkDown语法了,牛逼啊! [工匠若水 http://blog.csdn.net/yanbober] 本篇继续接上一篇,阅读上一篇EventBus使用之基础 背景 开始分析EventBus前可以下看下EventBus开源框架的工程目录结构: 从上图可以发现,其实EventBus的代码量不是很大,还是很方便入手分析的. 开始分析 通过上一篇基础使用可以发现,使用EventBus框架第一步是得到