詳解Guava中EventBus的使用
Guava EventBus
EventBus是Guava的事件處理機(jī)制,是設(shè)計(jì)模式中觀察者模式(生產(chǎn)/消費(fèi)者編程模型)的優(yōu)雅實(shí)現(xiàn)。對(duì)于事件監(jiān)聽和發(fā)布訂閱模式,EventBus使用非常簡(jiǎn)單便捷。

如果你做過(guò)CS的開發(fā),下面這段代碼可能會(huì)比較熟悉。
Button?button?=?new?Button("確定");
button.addListener(?new?Listener(){
????...
????public?void?onClick(Event?event){
????????//?
????}
????...
}?);
為按鈕注冊(cè)事件監(jiān)聽,當(dāng)按鈕被點(diǎn)擊時(shí),則觸發(fā)監(jiān)聽中相應(yīng)的回調(diào)。在上面的代碼中,有三個(gè)角色事件(Event),事件源(Button),監(jiān)聽(Listener),按鈕作為事件源,當(dāng)點(diǎn)擊行為觸發(fā)時(shí),會(huì)將該行為封裝成對(duì)應(yīng)的點(diǎn)擊事件,并根據(jù)行為類型將事件傳遞到響應(yīng)的監(jiān)聽器上, 這也就是我們常說(shuō)的監(jiān)聽器模式。
使用場(chǎng)景
實(shí)現(xiàn)消息生產(chǎn)者與消費(fèi)者間的解耦,對(duì)應(yīng)事件源與監(jiān)聽器,而消息則是事件
通過(guò)事件驅(qū)動(dòng)業(yè)務(wù)流程扭轉(zhuǎn),通過(guò)異步執(zhí)行機(jī)制實(shí)現(xiàn)代碼非阻塞執(zhí)行
擴(kuò)展主線外的分支業(yè)務(wù),減少代碼的侵入,比如各個(gè)環(huán)節(jié)的消息通知、短信提醒等
實(shí)現(xiàn)消息廣播到不同的模塊中
示例
訂單支付時(shí)的消息發(fā)送
//?商品
public?class?ProductOrder?{
????private?String?user;?//?用戶
????private?String?product;?//?商品
????private?double?amount;?//?金額
????@Override
????public?String?toString()?{
????????return?String.format("用戶:%s購(gòu)買了商品:%s,總金額:%s",?user,?product,?amount);
????}
}
//?事件
????@Data
????@AllArgsConstructor
????public?static?class?CreateOrderEvent?implements?OrderEvent{
????????private?ProductOrder?order;
????}
//?監(jiān)聽
????public?static?class?CreateOrderListener{
????????@Subscribe
????????public?void?onEvent(CreateOrderEvent?event)?{
????????????log.info("創(chuàng)建訂單:{}",?event.getOrder());
????????}
????}
測(cè)試: 我們可以定義各種事件,比如訂單創(chuàng)建、訂單取消、訂單支付... 只需要簡(jiǎn)單的三個(gè)步驟即可:
//?1.?創(chuàng)建事件總線 ????EventBus?eventBus?=?new?EventBus(?ProductOrder.class.getName()?); //?2.?注冊(cè)事件監(jiān)聽 ????eventBus.register(?new?CreateOrderListener()?); ????eventBus.register(?new?PayOrderListener()?); ????eventBus.register(?new?CancelOrderListener()?); ????eventBus.register(?new?RenewOrderListener()?); //?3.?發(fā)送事件通知 ????eventBus.post(new?ProductOrder.CreateOrderEvent(order)); ????TimeUnit.SECONDS.sleep(1); ????eventBus.post(new?ProductOrder.CancelOrderEvent(order)); ????TimeUnit.SECONDS.sleep(1); ????eventBus.post(new?ProductOrder.RenewOrderEvent(order)); ????TimeUnit.SECONDS.sleep(1); ????eventBus.post(new?ProductOrder.PayOrderEvent(order)); ????TimeUnit.SECONDS.sleep(5); ????eventBus.post(new?ProductOrder.ReturnOrderEvent(order));
同時(shí)我們可以通過(guò)AsyncEventBus建立事件異步總線,這樣在事件被觸發(fā)時(shí),可以異步通知監(jiān)聽者完成事件回調(diào),以此來(lái)提高響應(yīng)速度。
核心
EventBus
事件總線,可以理解為事件與監(jiān)聽器的上下文,主要實(shí)現(xiàn)事件的注冊(cè)、事件的分發(fā)、以及監(jiān)聽器的回調(diào),主要提供的方法包括:
- register 注冊(cè)監(jiān)聽,將監(jiān)聽器注冊(cè)到事件總線,通過(guò)注解@Subscribe通知其監(jiān)聽的事件類型(第一個(gè)方法參數(shù)類型)
- unregister 卸載監(jiān)聽,從事件總線移除監(jiān)聽
- post 發(fā)送事件通知,根據(jù)post事件類型,找到所有訂閱了該類型事件的監(jiān)聽器,并將事件推送到監(jiān)聽器對(duì)應(yīng)的監(jiān)聽方法
Subscribe
通過(guò)@Subscribe標(biāo)識(shí)監(jiān)聽器所關(guān)注的事件類型
Event
可以是任何對(duì)象,當(dāng)然不建議將基礎(chǔ)類型或String作為事件類型,這樣就沒法做到按類型區(qū)分了

通過(guò)上面的圖就可以很清楚各個(gè)各個(gè)組件的職責(zé),以及如何通過(guò)事件總線完成事件向監(jiān)聽的傳播,最終基于事件回調(diào)機(jī)制完成消息傳遞?;谑录?qū)動(dòng)的服務(wù)模型
上面這種結(jié)構(gòu)的圖形是不是在很多位置都見過(guò),這是一種經(jīng)典的設(shè)計(jì)模式。試想一下,我們不通過(guò)事件驅(qū)動(dòng)行為時(shí),一般你們?cè)趺磳懘a,通過(guò)ifelse?或者其他有著異曲同工的 實(shí)現(xiàn)方法,目的最后都是一樣。基于Guava提供的工具,我們不僅在使用時(shí)只需要簡(jiǎn)單的三個(gè)步驟就能實(shí)現(xiàn),同樣,當(dāng)需要屏蔽該功能時(shí)只需要去掉register一行即可,對(duì)整體功能 也沒有任何的影響。
在我們引入某種設(shè)計(jì)模式,某種架構(gòu)模型時(shí),總的目的都是為了降低代碼模塊間的耦合度,提升代碼整體的可讀性,最終讓代碼能夠易于維護(hù)性,或者有一定的復(fù)用性。
總結(jié)
事件監(jiān)聽模式、觀察者模式、發(fā)布訂閱模式,都是非常的相似,通過(guò)建立事件與監(jiān)聽器、觀察者與被觀察者、生產(chǎn)者與消費(fèi)者者間消息傳遞媒介(示例中的事件總線EventBus),
不僅能夠使消息的發(fā)起者與接收者之間進(jìn)行解耦,最主要的是通過(guò)消息傳遞渠道實(shí)現(xiàn)消息異步傳播,提升系統(tǒng)效率
到此這篇關(guān)于詳解Guava中EventBus的使用的文章就介紹到這了,更多相關(guān)Guava EventBus內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java使用Hutool執(zhí)行日期的加法和減法操作方法
使用Hutool進(jìn)行日期的加法和減法操作,可以使用`DateUtil.offsetXXX()`方法來(lái)實(shí)現(xiàn),這些方法會(huì)返回一個(gè)新的日期,而不是在原日期上進(jìn)行修改,本文給大家介紹Java使用Hutool執(zhí)行日期的加法和減法操作方法,感興趣的朋友一起看看吧2023-11-11
Springboot 整合通用mapper和pagehelper展示分頁(yè)數(shù)據(jù)的問(wèn)題(附github源碼)
這篇文章主要介紹了Springboot 整合通用mapper和pagehelper展示分頁(yè)數(shù)據(jù)(附github源碼),本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-09-09
WebSocket實(shí)現(xiàn)數(shù)據(jù)庫(kù)更新時(shí)前端頁(yè)面刷新
這篇文章主要為大家詳細(xì)介紹了WebSocket實(shí)現(xiàn)數(shù)據(jù)庫(kù)更新時(shí)前端頁(yè)面刷新,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-04-04
深入理解Spring注解@Async解決異步調(diào)用問(wèn)題
這篇文章主要介紹了深入理解Spring注解@Async解決異步調(diào)用問(wèn)題,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-07-07
利用spring aop實(shí)現(xiàn)動(dòng)態(tài)代理
這篇文章主要為大家詳細(xì)介紹了利用spring aop實(shí)現(xiàn)動(dòng)態(tài)代理的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-03-03
Spring?Boot中的過(guò)濾器攔截器監(jiān)聽器使用技巧匯總
本文將介紹在Spring?Boot應(yīng)用程序中使用過(guò)濾器、攔截器和監(jiān)聽器的使用技巧,我們將討論它們之間的區(qū)別,以及何時(shí)使用它們,我們還將提供代碼示例,以幫助您在自己的應(yīng)用程序中使用它們2023-12-12
基于Map的computeIfAbsent的使用場(chǎng)景和使用方式
這篇文章主要介紹了基于Map的computeIfAbsent的使用場(chǎng)景和使用方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09
SpringBoot使用不同環(huán)境動(dòng)態(tài)加載不同配置文件
通過(guò)在resource目錄下創(chuàng)建不同環(huán)境的配置文件,并在Spring?Boot啟動(dòng)類中使用環(huán)境變量來(lái)加載相應(yīng)的配置文件,從而實(shí)現(xiàn)不同環(huán)境下的配置自動(dòng)加載2024-11-11

