国产片侵犯亲女视频播放_亚洲精品二区_在线免费国产视频_欧美精品一区二区三区在线_少妇久久久_在线观看av不卡

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - ASP.NET教程 - 詳解ABP框架中領域層的領域事件Domain events

詳解ABP框架中領域層的領域事件Domain events

2020-01-15 13:42陽光銘睿 ASP.NET教程

ABP是基于ASP.NET框架之上的Web開發框架(GitHub:https://github.com/aspnetboilerplate),這篇我們來詳解ABP框架中領域層的領域事件Domain events,需要的朋友可以參考下

在C#中,一個類可以定義其專屬的事件并且其它類可以注冊該事件并監聽,當事件被觸發時可以獲得事件通知。這對于對于桌面應用程序或獨立的Windows Service來說非常有用。但是, 對于Web應用程序來說會有點問題,因為對象是根據請求(request)被創建并且它們的生命周期都很短暫。我們很難注冊其它類別的事件。同樣地,直接注冊其它類別的事件也造成了類之間的耦合性。

在應用系統中,領域事件被用于解耦并且重用(re-use)商業邏輯。

事件總線
事件總線為一個單體(singleton)的對象,它由所有其它類所共享,可通過它觸發和處理事件。要使用這個事件總線,你需要引用它。你可以用兩種方式來實現:

獲取默認實例( Getting the default instance)

你可以直接使用EventBus.Default。它是全局事件總線并且可以如下方式使用:

?
1
EventBus.Default.Trigger(...); //觸發事件

注入IEventBus事件接口(Injecting IEventBus)

除了直接使用EventBus.Default外,你還可以使用依賴注入(DI)的方式來取得IEventBus的參考。這利于進行單元測試。在這里,我們使用屬性注入的范式:

?
1
2
3
4
5
6
public class TaskAppService : ApplicaService {
 public IEventBus EventBus { get; set; }
 public TaskAppService() {
  EventBus = NullEventBus.Instance;
 }
}

注入事件總線,采用屬性注入比建構子注入更適合。事件是由類所描述并且該事件對象繼承自EventData。假設我們想要觸發某個事件于某個任務完成后:

?
1
2
3
public class TaskCompletedEventData : EventData {
 public int TaskId { get; set; }
}

這個類所包含的屬性都是類在處理事件時所需要的。EventData類定義了EventSource(那個對象觸發了這個事件)和EventTime(何時觸發)屬性。

定義事件
ABP定義AbpHandledExceptionData事件并且在異常發生的時候自動地觸發這個事件。這在你想要取得更多關于異常的信息時特別有用(即便ABP已自動地紀錄所有的異常)。你可以注冊這個事件并且設定它的觸發時機是在異常發生的時候。

ABP也提供在實體變更方面許多的通用事件數據類: EntityCreatedEventData, EntityUpdatedEventData和EntityDeletedEventData。它們被定義在Abp.Events.Bus.Entitis命名空間中。當某個實體新增/更新/刪除后,這些事件會由ABP自動地觸發。如果你有一個Person實體,可以注冊到EntityCreatedEventData,事件會在新的Person實體創建且插入到數據庫后被觸發。這些事件也支持繼承。如果Student類繼承自Person類,并且你注冊到EntityCreatedEventData中,接著你將會在Person或Student新增后收到觸發。

觸發事件
觸發事件的范例如下:

?
1
2
3
4
5
6
7
8
9
10
11
public class TaskAppService : ApplicationService {
 public IEventBus EventBus { get; set; }
 public TaskAppService() {
  EventBus = NullEventBus.Instance;
 }
 
 public void CompleteTask(CompleteTaskInput input) {
  //TODO: 已完成數據庫上的任務
  EventBus.Trigger(new TaskCompletedEventData { TaskId = 42 } );
 }
}

這里有一些觸發方法的重載:

?
1
2
3
EventBus.Trigger<TaskcompletedEventData>(new TaskCompletedEventData { TaskId = 42});
EventBus.Trigger(this, new TaskCompletedEventData { TaskId = 42 });
EventBus.Trigger(typeof(TaskCompletedEventData), this, new TaskCompletedEventData { TaskId = 42});

事件處理
要進行事件的處理,你應該要實現IEventHandler接口如下所示:

?
1
2
3
4
5
public class ActivityWriter : IEventHandler<TaskCompletedEventData>, ITransientDependency {
 public void HandleEvent(TaskCompletedEventData eventData) {
  WriteActivity("A task is completed by id = " + eventData.TaskId);
 }
}

EventBus已集成到依賴注入系統中。就如同我們在上例中實現ITransientDependency那樣,當TaskCompleted事件觸發,它會創建一個新的ActivityWriter類的實體并且調用它的HandleEvent方法,并接著釋放它。詳情請見依賴注入(DI)一文。

1.基礎事件的處理(Handling base events)

 EventBus支持事件的繼承。舉例來說,你可以創建TaskEventData以及兩個繼承類:TaskCompletedEventData和TaskCreatedEventData: 
 

?
1
2
3
4
5
6
7
8
9
10
11
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來處理這兩個事件:

?
1
2
3
4
5
6
7
8
9
public class ActivityWriter : IEventHandler<TaskEventData>, ITransientDependency {
 public void HandleEvent(TaskEventData eventData) {
  if(eventData is TaskCreatedEventData) {
  ...
  }else{
  ...
  }
 }
}

當然,你也可以實現IEventHandler來處理所有的事件,如果你真的想要這樣做的話(譯者注:作者不太建議這種方式)。

2.處理多個事件(Handling multiple events)

在單個處理器(handler)中我們可以可以處理多個事件。此時,你應該針對不同事件實現IEventHandler。范例如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
public class ActivityWriter :
 IEventHandler<TaskCompletedEventData>,
 IEventHandler<TaskCreatedEventData>,
 ITransientDependency
{
 public void HandleEvent(TaskCompletedEventData eventData) {
  //TODO: 處理事件
 }
 public void HandleEvent(TaskCreatedEventData eventData) {
  //TODO: 處理事件
 }
}

注冊處理器
我們必需注冊處理器(handler)到事件總線中來處理事件。

1.自動型Automatically

ABP掃描所有實現IEventHandler接口的類,并且自動注冊它們到事件總線中。當事件發生, 它通過依賴注入(DI)來取得處理器(handler)的引用對象并且在事件處理完畢之后將其釋放。這是比較建議的事件總線使用方式于ABP中。

2.手動型(Manually)

也可以通過手動注冊事件的方式,但是會有些問題。在Web應用程序中,事件的注冊應該要在應用程序啟動的時候。當一個Web請求(request)抵達時進行事件的注冊,并且反復這個行為。這可能會導致你的應用程序發生一些問題,因為注冊的類可以被調用多次。同樣需要注意的是,手動注冊無法與依賴注入系統一起使用。

ABP提供了多個事件總線注冊方法的重載(overload)。最簡單的一個重載方法是等待委派(delegate)或Lambda。

?
1
2
3
4
EventBus.Register<TaskCompletedEventData>(eventData =>
 {
  WriteActivity("A task is completed by id = " + eventData.TaskId);
 });

因此,事件:task completed會發生,而這個Lambda方法會被調用。第二個重載方法等待的是一個對象,該對象實現了IEventHandler:

?
1
Eventbus.Register<TaskCompletedEventData>(new ActivityWriter());

相同的例子,如果ActivityWriter因事件而被調用。這個方法也有一個非泛型的重載。另一個重載接受兩個泛化的參數:

EventBus.Register<TaskCompletedEventData, ActivityWriter>();
此時,事件總線創建一個新的ActivityWriter于每個事件。當它釋放的時候,它會調用ActivityWriter.Dispose方法。

最后,你可以注冊一個事件處理器工廠(event handler factory)來負責創建處理器。處理器工廠有兩個方法: GetHandler和ReleaseHandler,范例如下:

?
1
2
3
4
5
6
7
8
public class ActivityWriterFactory : IEventHandlerFactory {
  public IEventHandler GetHandler() {
   return new ActivityWriter();
  }
  public void ReleaseHandler(IEventHandler handler) {
   //TODO: 釋放ActivityWriter實體(處理器)
  }
 }

ABP也提供了特殊的工廠類,IocHandlerFactory,通過依賴注入系統,IocHandlerFactory可以用來創建或者釋放(dispose)處理器。ABP可以自動化注冊IocHandlerFactory。因此,如果你想要使用依賴注入系統,請直接使用自動化注冊的方式。

取消注冊事件
當你手動注冊事件總線,你或許想要在之后取消注冊。最簡單的取消事件注冊的方式即為registration.Dispose()。舉例如下:

?
1
2
3
4
//注冊一個事件
Var registration = EventBus.Register<TaskCompletedEventData>(eventData => WriteActivity("A task is completed by id = " + eventData.TaskId));
//取消注冊一個事件
registration.Dispose();

當然,取消注冊可以在任何地方任何時候進行。保存(keep)好注冊的對象并且在你想要取消注冊的時候釋放(dispose)掉它。所有注冊方法的重載(overload)都會返回一個可釋放(disposable)的對象來取消事件的注冊。

事件總線也提供取消注冊方法。使用范例:

?
1
2
3
4
5
6
//創建一個處理器
var handler = new ActivityWriter();
//注冊一個事件
EventBus.Register<TaskCompletedEventData>(handler);
//取消這個事件的注冊
EventBus.Unregister<TaskCompletedEventData>(handler);

它也提供重載的方法給取消注冊的委派和工廠。取消注冊處理器對象必須與之前注冊的對象是同一個。

最后,EventBus提供一個UnregisterAll()方法來取消某個事件所有處理器的注冊,而UnregisterAll()方法則是所有事件的所有處理器。

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 一a级毛片 | 国产一区二区三区在线 | 欧美精品在欧美一区二区少妇 | 精品国产不卡一区二区三区 | 成人免费黄色片 | 欧美亚洲视频 | 国产日韩精品一区 | 国产亚洲欧美一区二区 | 最近免费中文字幕大全免费版视频 | 午夜成人免费影院 | 国产精品自产拍在线观看桃花 | 欧美中文字幕一区 | 亚洲精品视频免费在线观看 | 日本一区二区三区免费观看 | 五月婷婷激情网 | 成年人在线观看免费视频 | 国产精品区二区三区日本 | 女教师高潮叫床视频在线观看 | 亚洲成人av一区二区三区 | 福利片网站 | 五月激情综合 | 亚洲国产精品久久久久婷婷老年 | 精品中文字幕在线 | 亚洲精品久久久久久久久久久 | 欧美色欧美亚洲另类七区 | 成人免费毛片aaaaaa片 | 亚洲a网 | 最新国产精品精品视频 | 青青草原亚洲 | 国产亚洲精品美女久久久久久久久久 | 91视频免费| 韩日中文字幕 | 欧州一区二区三区 | 欧美在线视频网站 | 久久久精品日本 | 亚洲一级毛片 | 青娱乐网 | 日韩国产欧美视频 | 久久久久久亚洲精品 | 日韩三级在线 | 黄a在线观看 |