在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

為什么需要Repository?什么才是好的Repository ?

jf_ro2CN3Fa ? 來源:geekhalo ? 2023-03-07 09:11 ? 次閱讀

1. 領域事件

領域事件是 DDD 中重要的模式之一,主要用于模型或系統間的解耦,提高系統的可擴展性和可維護性。

1.1. 什么是領域事件

領域事件是領域驅動設計(Domain-Driven Design,簡稱DDD)中的一個重要概念,特指在領域模型中發生的有意義的事件,是對領域模型中的重要業務動作執行結果的抽象,如訂單創建、支付完成等。

在DDD中,領域事件是一種用于傳遞信息的機制,它使得不同領域模型之間的通信變得更加簡單和靈活。通過將事件分發給相關的訂閱者,可以讓不同的領域模型之間實現松耦合,從而更容易擴展和維護應用程序。

領域事件通常由領域對象主動觸發并發布,而事件處理器則負責訂閱事件并對事件進行處理。通過事件發布和訂閱機制,可以在應用程序中實現高效的事件驅動架構,從而更好地支持復雜的業務邏輯和業務流程。

說起來有點抽象,簡單舉個例子:假設有一個電子商務系統,用戶下單后需要生成訂單并發送通知給相關人員。在領域模型中,可以定義一個 Order 領域對象,該對象可以包含多個屬性,如訂單號、下單時間、購買的商品信息、收貨地址等等。當用戶下單時,可以通過調用 Order 對象的方法來生成訂單,同時也可以通過領域事件來發送通知。

具體來說,可以定義一個 OrderCreated 領域事件,用于表示訂單創建完成的事件,該事件包含一些必要的屬性,如訂單號、下單時間、購買的商品信息、收貨地址等等。當 Order 對象創建完成后,可以通過領域事件來觸發發送通知的操作,比如發送郵件或短信通知相關人員。

1.2. 領域事件的應用場景

領域事件的應用創建眾多,從圖中可以看出:

04b8ee02-bc7e-11ed-bfe3-dac502259ad0.png

領域事件可以:

保證聚合間的數據一致性。當一個聚合根上的操作引發了其他聚合根的變更時,將這些變更作為領域事件發布出去,其他聚合根可以訂閱這些事件并更新自己的狀態,從而實現最終一致性。

替換批量處理。可以作為任務的觸發器,例如定時任務、異步任務,避免定時+掃描這類批量處理。

實現事件源模式。將所有的領域事件全部存儲下來,可以用于恢復聚合的狀態,實現事件源模式;也可以用于后續的審計和調試。

進行限界上下文集成。將事件從一個子域發布到另一個子域,使得這兩個子域可以解耦,不用相互知道彼此的存在。

領域事件雖好,但仍需技術框架進行支持,其實 Spring 的 Event 機制就足以滿足各類需求。

2. Spring 對 Event 的支持

在 Spring 中,事件的處理可以通過三種方式來實現:

基于接口的事件處理:通過實現 ApplicationListener 接口并重寫 onApplicationEvent 方法來處理事件。

基于注解的事件處理:通過在方法上添加 @EventListener 或 @TransactionEventListener 注解來處理事件,可以指定事件的類型以及監聽的條件等。

基于異步事件處理:通過使用 @Async 注解來異步處理事件,可以提高應用程序的響應速度。

2.1. 基于接口的事件處理

由于與 Spring 存在強耦合,現在已經很少使用,可以直接跳過。

下面是一個基于接口的事件處理的示例代碼:

@Component
publicclassMyEventListenerimplementsApplicationListener{
@Override
publicvoidonApplicationEvent(MyEventevent){
//處理事件
System.out.println("Receivedevent:"+event.getMessage());
}
}

publicclassMyEvent{
privateStringmessage;

publicMyEvent(Stringmessage){
this.message=message;
}

publicStringgetMessage(){
returnmessage;
}
}

@Component
publicclassMyEventPublisher{
@Autowired
privateApplicationEventPublishereventPublisher;

publicvoidpublishEvent(Stringmessage){
MyEventevent=newMyEvent(message);
eventPublisher.publishEvent(event);
}
}

在這個示例中,MyEvent 是一個自定義的事件類,MyEventListener 是一個實現了 ApplicationListener 接口的監聽器,用于處理 MyEvent 事件,MyEventPublisher 是用于發布事件的類。

當應用程序調用 MyEventPublisher 的 publishEvent 方法時,會觸發一個 MyEvent 事件,MyEventListener 中的 onApplicationEvent 方法將被自動調用,從而處理這個事件。

2.2. 基于注解的事件處理

Spring 提供 @EventListener 和 @TransactionListener 兩個注解以簡化對事件的處理。

2.2.1. @EventListener

Spring 的 EventListener 監聽器是一種相對于傳統的事件監聽方式更為簡潔和靈活的事件機制。與傳統的事件機制不同,EventListener 不需要顯示地繼承特定的事件接口,而是使用注解標識需要監聽的事件類型,然后通過一個單獨的監聽器類處理所有類型的事件。

相比之下 EventListener 的優勢主要有以下幾點:

更加靈活:EventListener 不依賴于任何特定的事件接口,從而使得事件處理更加靈活,可以監聽和處理任意類型的事件。

更加簡潔:相比傳統的事件監聽方式,使用 EventListener 可以避免一系列繁瑣的接口定義和實現,簡化了代碼結構,使得開發效率更高。

更加松耦合:EventListener 將事件發布方和事件處理方分離,遵循松耦合的設計原則,提高了代碼的可維護性和擴展性。

更加可測試:由于 EventListener 可以監聽和處理任意類型的事件,可以通過單元測試驗證其功能是否正確,從而提高了測試的可靠性。

以下是一個簡單的例子:

@Component
publicclassMyEventListener{

@EventListener
publicvoidonApplicationEvent(MyEventevent){
//處理事件
System.out.println("Receivedevent:"+event.getMessage());
}
}

publicclassMyEvent{
privateStringmessage;

publicMyEvent(Stringmessage){
this.message=message;
}

publicStringgetMessage(){
returnmessage;
}
}

@Component
publicclassMyEventPublisher{

@Autowired
privateApplicationEventPublishereventPublisher;

publicvoidpublishEvent(Stringmessage){
MyEventevent=newMyEvent(message);
eventPublisher.publishEvent(event);
}
}

相比基于接口的事件處理,EventListener 是一種更加簡潔、靈活、松耦合、可測試的事件機制,能夠有效地降低開發的復雜度,提高開發效率。

2.2.2. @TransactionEventListener

在 Spring 中,TransactionEventListner 和 EventListner 都是用于處理事件的接口。不同之處在于

TransactionEventListner 是在事務提交后才會觸發

而 EventListner 則是在事件發布后就會觸發。

具體來說,在使用 Spring 的聲明式事務時,可以在事務提交后觸發某些事件。這就是 TransactionEventListner 的應用場景。而 EventListner 則不涉及事務,可以用于在事件發布后觸發一些操作。

下面是一個簡單的示例,演示了如何使用 TransactionEventListner 和 EventListner:

@Component
publicclassMyEventListener{

@EventListener
publicvoidhandleMyEvent(MyEventevent){
//處理MyEvent
}

@TransactionalEventListener
publicvoidhandleMyTransactionalEvent(MyTransactionalEventevent){
//處理MyTransactionalEvent
}
}

@Service
publicclassMyService{

@Autowired
privateApplicationEventPublishereventPublisher;

@Autowired
privateMyRepositorymyRepository;

@Transactional
publicvoiddoSomething(){
//做一些事情
MyEntityentity=myRepository.findById(1L);
//發布事件
eventPublisher.publishEvent(newMyEvent(this,entity));
//發布事務事件
eventPublisher.publishEvent(newMyTransactionalEvent(this,entity));
}
}

在這個例子中,MyEventListener 類定義了兩個方法,handleMyEvent 和 handleMyTransactionalEvent,分別處理 MyEvent 和 MyTransactionalEvent 事件。其中,handleMyTransactionalEvent 方法用 @TransactionalEventListener 注解標記,表示它只會在事務提交后觸發。

MyService 類中的 doSomething 方法使用 ApplicationEventPublisher 來發布事件。注意,它發布了兩種不同類型的事件:MyEvent 和 MyTransactionalEvent。這兩個事件會分別觸發 MyEventListener 中的對應方法。

總的來說,Spring 的事件機制非常靈活,可以方便地擴展應用程序的功能。TransactionEventListner 和 EventListner 這兩個接口的應用場景有所不同,可以根據實際需求選擇使用。

2.3.基于異步事件處理

@Async是Spring框架中的一個注解,用于將一個方法標記為異步執行。使用該注解,Spring將自動為該方法創建一個新線程,使其在后臺異步執行,不會阻塞主線程的執行。

在具體應用中,使用@Async可以大大提升應用的并發處理能力,使得系統能夠更快地響應用戶請求,提高系統的吞吐量。

@Async 和 @EventListener 或 @TransactionEventListener 注解在一起使用時,會產生異步的事件處理器。使用這種組合的方式,事件處理器會在單獨的線程池中執行,以避免阻塞主線程。這種方式在需要處理大量事件或者事件處理器耗時較長的情況下非常有用,可以有效提高應用的性能和可伸縮性。同時,Spring 框架對這種方式也提供了完善的支持,可以方便地使用這種方式來實現異步事件處理。

下面是一個簡單的示例代碼,演示了如何在 Spring 中使用 @Async 和 @EventListener 一起實現異步事件處理:

@Component
publicclassExampleEventListener{

@Async
@EventListener
publicvoidhandleExampleEvent(ExampleEventevent){
//在新的線程中執行異步邏輯
//...
}
}

在這個示例中,ExampleEventListener 類中的 handleExampleEvent 方法使用了 @Async 和 @EventListener 注解,表示這個方法是一個異步事件監聽器。當一個 ExampleEvent 事件被觸發時,這個方法會被異步地執行。在這個方法中,可以執行任何異步的邏輯處理,比如向隊列發送消息、調用其他服務等。

備注:在使用 @Async 時,需要根據業務場景對線程池進行自定義,以免出現資源不夠的情況(Spring 默認使用單線程處理@Async異步任務)

4. 場景分析

綜上所述,當領域事件發出來之后,不同的注解會產生不同的行為,簡單匯總如下:

@EventListener @TransactionEventListener
無 @Async 順序、同步執行 事務提交后、同步執行
有 @Async 順序、異步執行 事務提交后、異步執行

4.1. @EventListener

04e6afea-bc7e-11ed-bfe3-dac502259ad0.png

特點:

順序執行。調用 publish(Event) 后,自動觸發對 @EventListner 注釋方法的調用

同步執行。使用主線程執行,方法拋出異常會中斷調用鏈路,會觸發事務的回歸

應用場景:

事務消息表。在同一事務中完成對業務數據和消息表的修改

業務驗證。對業務對象進行最后一次驗證,如果驗證不通過直接拋出異常中斷數據庫事務

業務插件。在當前線程和事務中執行插件完成業務擴展

4.2. @TransactionEventListener

04f892e6-bc7e-11ed-bfe3-dac502259ad0.png

特點:

事務提交后執行。調用 publish(Event) 時,只是向上下文中注冊了一個回調器,并不會立即執行;只有在事務提交后,才會觸發對 @TransactionEventListner 注釋方法的調用

同步執行。使用主線程執行,方法拋出異常會中斷調用鏈路,當不會回歸事務(事務已提交,沒有辦法進行回歸)

應用場景:

數據同步。事務提交后,將變更同步到 ES 或 Cache

記錄審計日志。只有在業務變更成功更新到數據庫時才進行記錄

備注:@TransactionEventLisnter 必須在事務上下文中,脫離上下文,調用不會生效

4.3. @EventListener + @Async

0509c2b4-bc7e-11ed-bfe3-dac502259ad0.png

特點:

順序執行。調用 publish(Event) 后,自動觸發對 @EventListner 注釋方法的調用

異步執行。使用獨立的線程池執行任務,方法拋出異常對主流程沒有任何影響

應用場景:

記日志明細日志,輔助排查問題

4.4. @TransactionEventListener + @Async

051f67cc-bc7e-11ed-bfe3-dac502259ad0.png

特點:

事務提交后執行。調用 publish(Event) 時,只是向上下文中注冊了一個回調器,并不會立即執行;只有在事務提交后,才會觸發對 @TransactionEventListner 注釋方法的調用

異步執行。使用獨立的線程池執行任務,方法拋出異常對主流程沒有任何影響

應用場景:異步處理。記錄操作日志,異步保存數據等 備注:@TransactionEventLisnter 必須在事務上下文中,脫離上下文,調用不會生效

5. 小結

領域事件的落地,不僅需要強大的設計能力,還需要與之匹配的基礎設施。Spring 作為最常用的框架,基于發布訂閱實現了完整的一套 Event 管理機制。工具在手是否能根據業務場景選擇合適的解決方案就成了研發的職責,簡單思考以下組合適用場景是什么:

@EventListener

@TransactionEventListener

@EventListener + @Async

@TransactionEventListener + @Async






審核編輯:劉清

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • JAVA
    +關注

    關注

    19

    文章

    2973

    瀏覽量

    104926
  • MySQL
    +關注

    關注

    1

    文章

    825

    瀏覽量

    26657
  • MYSQL數據庫
    +關注

    關注

    0

    文章

    96

    瀏覽量

    9416
  • ddd
    ddd
    +關注

    關注

    0

    文章

    23

    瀏覽量

    2935

原文標題:Spring Event + DDD = 王炸!!

文章出處:【微信號:芋道源碼,微信公眾號:芋道源碼】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    modusToolbox在加載離線庫時,總是在loading device_db時提示錯誤的原因?

    /https___github.com_Infineon_device-db/device-db\' does not appear to be a git repository fatal: Could not read from
    發表于 02-02 08:07

    使用MotorControl Workbench生成代碼時出錯的原因?

    各位大師你們!使用MotorControl Workbench生成代碼時出錯,請高手指教,謝謝。 Code generation started CMSIS Pack version
    發表于 04-18 07:08

    How to download and install AD10 when using proxy servers

    create your own installation repository. This provides a local install solution for computers
    發表于 09-05 10:54

    MATLAB請教各位大神

    ModelAdvisor.Repository/connect (line 14)出錯 ModelAdvisor.Repository (line 14)出錯 Simulink.ModelAdvisor/getWorkDir
    發表于 10-24 18:42

    插牌具有什么功能才是需要的?

    插牌具有什么功能才是需要的?
    發表于 07-13 13:57

    ESP-IDF使用:請問怎么樣能測試esp32的spp example

    各位大佬指點迷津。$ makefatal: Not a git repository (or any of the parent directories): .gitfatal: Not a git
    發表于 08-17 09:22

    STM32CubeMX芯片包(固件庫)使用注意事項

    1. 解壓本地固件庫包后需要將文件夾放到指定路徑,該指定路徑為cubemx軟件中的help -> updater settings -> updater settings ->
    發表于 07-23 09:03

    STM32CubeIDE不顯示“Inizialize all peripherals with their Default Mode”怎么解決?

    大家,我有這個痛苦的問題:當我想創建一個新項目時,選擇板子和項目名稱后點擊“完成”并沒有顯示可以選擇加載外設默認模式的窗口,而是立即加載空白配置。我試過其他版本的多維數據集并刪除“\Repository”但沒有任何改變。請問有什么建議嗎?
    發表于 01-03 09:50

    STM32CubeMX系列6版本刪除主文件夾中存在的所有文件夾,從而刪除用戶創建的文件夾要如何避免?

    我將 STM32CubeMX 安裝到文件夾C:\STM32CubeMX,然后將 REPOSITORY FOLDER 設置為C:\STM32CubeMX\Repository,因為我想將
    發表于 01-13 08:32

    如何在Touchgfx設計器中開始使用來自touchgfx-open-repository的圖形小部件?

    代碼讓它們做一些事情。我看不到添加圖表的方法,所以我用谷歌搜索了一下,發現您需要來自此處存儲庫的圖表小部件:https ://github.com/touchgfx
    發表于 01-30 08:09

    如何在代碼中從touchgfx-open-repository添加二維碼小部件?

    ) 、 qrCode.setQRCode(&code) 、 qrCode.setScale(4) 、 add(qrCode) 方法,但我找不到它。https://github.com/touchgfx/touchgfx-open-repository/tree/master/widgets/QRCode你能幫忙嗎?
    發表于 02-08 07:44

    【米爾MYD-JX8MMA7開發板-ARM+FPGA架構試用體驗】快速入門

    : The repository 'http://ftp.debian.org/debian buster InRelease' is not signed.N: Updating from such a
    發表于 03-09 19:04

    mirror和repository的區別分析

    internal repository是指在局域網內部搭建的repository,它跟central repository, jboss repository等的區別僅僅在于其URL是
    發表于 11-29 10:42 ?4251次閱讀
    mirror和<b class='flag-5'>repository</b>的區別分析

    怎么樣才是一臺電腦

    怎么樣才是一臺電腦?即使是天天進行測試的的編輯,都要不斷地面對和解決這個問題。性能,穩定性,兼容性,平臺能力,都對都很重要,而單獨強調某一個側面卻都不夠全面。
    發表于 05-17 14:46 ?1697次閱讀

    pip安裝更換鏡像

    使用pip來安裝python包有時候安裝起來會非常慢,因此需要換成國內的源來加速下載: 使用命令 以Torch為例: pip install -i https
    的頭像 發表于 01-11 14:54 ?632次閱讀
    主站蜘蛛池模板: 天天上天天操| 一级黄色录像视频| 欧美成人高清性色生活| 欧美成人生活片| 黄色日本网站| 国产哺乳期奶水avav| www.色亚洲| 特黄十八岁大片| 国产精品午夜免费观看网站| 网全大全黄| 亚洲a在线观看| 色婷婷激婷婷深爱五月小说| 欧美人与禽| 国产精品视频色拍拍| 欧美婷婷六月丁香综合色| 鲁老汉精品视频在线观看| 国产精品日本亚洲777| 一级特黄aaa大片| 欧美成人性色生活片天天看| 亚洲mm8成为人影院| 亚洲黄色在线网站| 亚洲欧美日韩国产一区二区三区精品| 乱人伦的小说| 午夜精品在线免费观看| 女人张开腿让男人捅爽| 9久久9久久精品| 99色综合| 免费高清特黄a 大片| 91pao强力打造免费高清| 亚洲色图88| 免费能看的黄色网址| 在线天堂资源www中文在线| 国产自在自线午夜精品视频在| 中文字幕一区二区三区在线播放| 亚洲精品456| 免费人成在观看| 午夜禁片| 国产精品乳摇在线播放| 久久综合色播| 狠狠色噜噜狠狠狠狠97不卡| 人人做人人澡人人人爽|