Spring中ApplicationEventPublisher發(fā)布訂閱模式的實(shí)現(xiàn)
概念
關(guān)于發(fā)布訂閱這個(gè)詞,其實(shí)不僅僅出現(xiàn)在Spring框架當(dāng)中,其實(shí)在Redis中也有存在(其對(duì)應(yīng)的是convertAndSend()方法),還有在MQ消息隊(duì)列里也是有的,但這里就主要介紹的是關(guān)于Spring框架的ApplicationEventPublisher如何做到消息的發(fā)布與訂閱。隨著現(xiàn)在的業(yè)務(wù)量和需求量越來越大,其實(shí)基本都是分布式微服務(wù)集群的使用了,所以基本都是用到Redis與MQ。但是對(duì)于單體Spring Boot應(yīng)用時(shí),用Spring自帶的發(fā)布訂閱就已經(jīng)綽綽有余了。
需要知道的是發(fā)布訂閱需要有三個(gè)對(duì)象:事件、事件發(fā)布源、事件接收源(監(jiān)聽器)。
事件
對(duì)于事件來說,需要去繼承ApplicationEvent。至于為什么需要繼承ApplicationEvent就需要研究源碼,等我后續(xù)看完再回來更新筆記。
public class LogEvent extends ApplicationEvent { public LogEvent(Object message) { super(message); } }
事件發(fā)布源
這個(gè)基本上都是業(yè)務(wù)層的代碼,但是作為學(xué)習(xí)階段,我將其直接放在控制層。關(guān)于如何去發(fā)布,就需要用到Spring中的ApplicationEventPublisher。
@Slf4j @RestController @RequestMapping("/log") public class LogController { @Autowired private ApplicationEventPublisher publisher; @GetMapping("/publisher") public void log() { log.info("進(jìn)入到log方法,開始發(fā)送事件"); publisher.publishEvent(new LogEvent("log方法生成事件信息")); log.info("log方法發(fā)送事件完畢"); } }
監(jiān)聽器
/** * 管理員日志監(jiān)聽器 */ @Slf4j @Component public class AdminListener { @EventListener(LogEvent.class) public void adminListen(LogEvent logEvent) { try { log.info("管理員監(jiān)聽到的日志信息為,{}", logEvent); Thread.sleep(5000); // 睡眠5s log.info("管理員監(jiān)聽完畢"); } catch (InterruptedException e) { e.printStackTrace(); } } }
/** * 用戶日志事件監(jiān)聽器 */ @Slf4j @Component public class UserListener { @EventListener(LogEvent.class) public void userListen(LogEvent logEvent) { try { log.info("用戶監(jiān)聽到的日志信息為,{}", logEvent); Thread.sleep(10000); // 睡眠10s log.info("用戶監(jiān)聽完畢"); } catch (InterruptedException e) { e.printStackTrace(); } } }
執(zhí)行結(jié)果
從執(zhí)行結(jié)果可以發(fā)現(xiàn)管理員線程睡眠了5s后執(zhí)行完畢才會(huì)執(zhí)行用戶的代碼塊。所以表明是同步執(zhí)行的。
于是為了將同步執(zhí)行變?yōu)楫惒綀?zhí)行,又在兩個(gè)監(jiān)聽器的方法上添加了@Async注解,并且一定要在啟動(dòng)類上添加@EnableAsync注解。再次執(zhí)行會(huì)發(fā)現(xiàn)兩次的執(zhí)行結(jié)果并不一致。兩個(gè)監(jiān)聽器是異步執(zhí)行的。甚至在監(jiān)聽器上放置@Order注解可以實(shí)現(xiàn)先后順序,但這里就不演示了。
源碼追蹤
這里只進(jìn)行粗略的源碼查看,因?yàn)樽髡吖ατ邢逕o法解釋得完全仔細(xì),以后看懂了再回來更新筆記??梢园l(fā)現(xiàn)其已經(jīng)使用了JDK8新特性了。從ApplicationEventPublisher的publisher(Application event)方法就會(huì)進(jìn)入到其子類AbstartApplicationContext中。
protected void publishEvent(Object event, @Nullable ResolvableType eventType) { Assert.notNull(event, "Event must not be null"); Object applicationEvent; if (event instanceof ApplicationEvent) { applicationEvent = (ApplicationEvent)event; } else { applicationEvent = new PayloadApplicationEvent(this, event); if (eventType == null) { eventType = ((PayloadApplicationEvent)applicationEvent).getResolvableType(); } } if (this.earlyApplicationEvents != null) { this.earlyApplicationEvents.add(applicationEvent); } else { // 在這步完成真正的發(fā)布訂閱 只有監(jiān)聽完才會(huì)向下繼續(xù)執(zhí)行 this.getApplicationEventMulticaster().multicastEvent((ApplicationEvent)applicationEvent, eventType); } if (this.parent != null) { if (this.parent instanceof AbstractApplicationContext) { ((AbstractApplicationContext)this.parent).publishEvent(event, eventType); } else { this.parent.publishEvent(event); } }
到此這篇關(guān)于Spring中ApplicationEventPublisher發(fā)布訂閱模式的實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)Spring ApplicationEventPublisher發(fā)布訂閱模式內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringBoot3.3中實(shí)現(xiàn)多端口監(jiān)聽的示例代碼
在SpringBoot應(yīng)用中實(shí)現(xiàn)多端口監(jiān)聽,可以讓一個(gè)應(yīng)用處理不同類型的HTTP請(qǐng)求或暴露多個(gè)服務(wù)接口,本文詳細(xì)講解了通過配置application.yml文件和編寫自定義配置類的方法,實(shí)現(xiàn)了對(duì)多個(gè)端口的監(jiān)聽,感興趣的可以了解一下2024-11-11淺談PrintStream和PrintWriter的區(qū)別和聯(lián)系
這篇文章主要介紹了淺談PrintStream和PrintWriter的區(qū)別和聯(lián)系,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-03-03Springboot詳解RocketMQ實(shí)現(xiàn)廣播消息流程
RocketMQ作為一款純java、分布式、隊(duì)列模型的開源消息中間件,支持事務(wù)消息、順序消息、批量消息、定時(shí)消息、消息回溯等,本篇我們了解如何實(shí)現(xiàn)廣播消息2022-06-06Java二分查找算法與數(shù)組處理的應(yīng)用實(shí)例
二分查找法,又叫做折半查找法,它是一種效率較高的查找方法。數(shù)組對(duì)于每一門編程語言來說都是重要的數(shù)據(jù)結(jié)構(gòu)之一,當(dāng)然不同語言對(duì)數(shù)組的實(shí)現(xiàn)及處理也不盡相同。Java 語言中提供的數(shù)組是用來存儲(chǔ)固定大小的同類型元素2022-07-07使用Spring Boot實(shí)現(xiàn)操作數(shù)據(jù)庫的接口的過程
本文給大家分享使用Spring Boot實(shí)現(xiàn)操作數(shù)據(jù)庫的接口的過程,包括springboot原理解析及實(shí)例代碼詳解,感興趣的朋友跟隨小編一起看看吧2021-07-07Java實(shí)現(xiàn)產(chǎn)生隨機(jī)字符串主鍵的UUID工具類
這篇文章主要介紹了Java實(shí)現(xiàn)產(chǎn)生隨機(jī)字符串主鍵的UUID工具類,涉及java隨機(jī)數(shù)與字符串遍歷、轉(zhuǎn)換等相關(guān)操作技巧,需要的朋友可以參考下2017-10-10