<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

          Spring Event,賊好用的業(yè)務(wù)解耦神器!

          共 6900字,需瀏覽 14分鐘

           ·

          2022-07-31 15:12

          程序員的成長之路
          互聯(lián)網(wǎng)/程序員/技術(shù)/資料共享 
          關(guān)注


          閱讀本文大概需要 5 分鐘。

          作者:九七年生于初夏
          來源:blog.csdn.net/csp732171109/article/details/124377254

          # 前言


          實(shí)際業(yè)務(wù)開發(fā)過程中,業(yè)務(wù)邏輯可能非常復(fù)雜,核心業(yè)務(wù) + N個子業(yè)務(wù)。如果都放到一塊兒去做,代碼可能會很長,耦合度不斷攀升,維護(hù)起來也麻煩,甚至頭疼。還有一些業(yè)務(wù)場景不需要在一次請求中同步完成,比如郵件發(fā)送、短信發(fā)送等。


          MQ 確實(shí)可以解決這個問題,但 MQ 重啊,非必要不提升架構(gòu)復(fù)雜度。針對這些問題,我們了解一下 Spring Event。

          # Spring Event 同步使用


          Spring Event(Application Event)其實(shí)就是一個觀察者設(shè)計模式,一個 Bean 處理完成任務(wù)后希望通知其它 Bean 或者說一個 Bean 想觀察監(jiān)聽另一個Bean 的行為。

          Spring Event 用來解耦業(yè)務(wù)真的賊好用!

          Demo 地址:


          https://gitee.com/csps/mingyue-springboot-learning

          1.自定義事件


          定義事件,繼承 ApplicationEvent 的類成為一個事件類
          import lombok.Data;import lombok.ToString;import org.springframework.context.ApplicationEvent;/** * @author Strive * @date 2022/4/22 18:00 * @description */@Data@ToStringpublic class OrderProductEvent extends ApplicationEvent { /** 該類型事件攜帶的信息 */  private String orderId;
          public OrderProductEvent(Object source, String orderId) { super(source); this.orderId = orderId; }}

          2.定義監(jiān)聽器

          監(jiān)聽并處理事件,實(shí)現(xiàn) ApplicationListener 接口或者使用 @EventListener 注解
          import com.csp.mingyue.event.events.OrderProductEvent;import lombok.SneakyThrows;import lombok.extern.slf4j.Slf4j;import org.springframework.context.ApplicationListener;import org.springframework.stereotype.Component;/** * 實(shí)現(xiàn) ApplicationListener 接口,并指定監(jiān)聽的事件類型 * * @author Strive * @date 2022/4/24 09:09 * @description */@Slf4j@Componentpublic class OrderProductListener implements ApplicationListener<OrderProductEvent> { /** 使用 onApplicationEvent 方法對消息進(jìn)行接收處理 */ @SneakyThrows @Override public void onApplicationEvent(OrderProductEvent event) { String orderId = event.getOrderId(); long start = System.currentTimeMillis(); Thread.sleep(2000); long end = System.currentTimeMillis(); log.info("{}:校驗(yàn)訂單商品價格耗時:({})毫秒", orderId, (end - start)); }}

          3.定義發(fā)布者


          發(fā)布事件,通過 ApplicationEventPublisher 發(fā)布事件
          import com.csp.mingyue.event.events.OrderProductEvent;import lombok.RequiredArgsConstructor;import lombok.extern.slf4j.Slf4j;import org.springframework.context.ApplicationContext;import org.springframework.stereotype.Service;
          /** * @author Strive * @date 2022/4/24 09:25 * @description */@Slf4j@Service@RequiredArgsConstructorpublic class OrderService { /** 注入ApplicationContext用來發(fā)布事件 */ private final ApplicationContext applicationContext; /** * 下單 * * @param orderId 訂單ID */ public String buyOrder(String orderId) { long start = System.currentTimeMillis(); // 1.查詢訂單詳情 // 2.檢驗(yàn)訂單價格 (同步處理) applicationContext.publishEvent(new OrderProductEvent(this, orderId)); // 3.短信通知(異步處理) long end = System.currentTimeMillis(); log.info("任務(wù)全部完成,總耗時:({})毫秒", end - start); return "購買成功"; }}

          4.單測執(zhí)行
          import org.junit.jupiter.api.Test;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.boot.test.context.SpringBootTest;
          /** * @author Strive * @date 2022/4/24 09:28 * @description */@SpringBootTestpublic class OrderServiceTest {  @Autowired private OrderService orderService;
          @Test public void buyOrderTest() { orderService.buyOrder("732171109"); }}

          執(zhí)行結(jié)果如下:
          2022-04-24 10:13:17.535 INFO 44272 --- [ main] c.c.m.e.listener.OrderProductListener : 732171109:校驗(yàn)訂單商品價格耗時:(2008)毫秒2022-04-24 10:13:17.536 INFO 44272 --- [ main] c.c.mingyue.event.service.OrderService : 任務(wù)全部完成,總耗時:(2009)毫秒

          # Spring Event 異步使用


          有些業(yè)務(wù)場景不需要在一次請求中同步完成,比如郵件發(fā)送、短信發(fā)送等。

          1.自定義事件
          import lombok.AllArgsConstructor;import lombok.Data;/** * @author Strive * @date 2022/4/24 10:18 * @description */@Data@AllArgsConstructorpublic class MsgEvent {
          /** 該類型事件攜帶的信息 */ public String orderId;}

          2.定義監(jiān)聽器


          推薦使用 @EventListener 注解
          import com.csp.mingyue.event.events.MsgEvent;import lombok.SneakyThrows;import lombok.extern.slf4j.Slf4j;import org.springframework.context.event.EventListener;import org.springframework.stereotype.Component;/** * @author Strive * @date 2022/4/24 10:20 * @description */@Slf4j@Componentpublic class MsgListener {
          @SneakyThrows @EventListener(MsgEvent.class) public void sendMsg(MsgEvent event) { String orderId = event.getOrderId(); long start = System.currentTimeMillis(); log.info("開發(fā)發(fā)送短信"); log.info("開發(fā)發(fā)送郵件"); Thread.sleep(4000); long end = System.currentTimeMillis(); log.info("{}:發(fā)送短信、郵件耗時:({})毫秒", orderId, (end - start)); }}

          3.定義發(fā)布者
          /** * 下單 * * @param orderId 訂單ID */public String buyOrder(String orderId) { long start = System.currentTimeMillis(); // 1.查詢訂單詳情
          // 2.檢驗(yàn)訂單價格 (同步處理)    applicationContext.publishEvent(new OrderProductEvent(this, orderId));
          // 3.短信通知(異步處理)    applicationContext.publishEvent(new MsgEvent(orderId));
          long end = System.currentTimeMillis(); log.info("任務(wù)全部完成,總耗時:({})毫秒", end - start); return "購買成功";}

          4.單測執(zhí)行(同步)
          @Testpublic void buyOrderTest() { orderService.buyOrder("732171109");}

          執(zhí)行結(jié)果如下:
          2022-04-24 10:24:13.905 INFO 54848 --- [ main] c.c.m.e.listener.OrderProductListener : 732171109:校驗(yàn)訂單商品價格耗時:(2004)毫秒2022-04-24 10:24:13.906 INFO 54848 --- [ main] c.c.mingyue.event.listener.MsgListener : 開發(fā)發(fā)送短信2022-04-24 10:24:13.907 INFO 54848 --- [ main] c.c.mingyue.event.listener.MsgListener : 開發(fā)發(fā)送郵件2022-04-24 10:24:17.908 INFO 54848 --- [ main] c.c.mingyue.event.listener.MsgListener : 732171109:發(fā)送短信、郵件耗時:(4002)毫秒2022-04-24 10:24:17.908 INFO 54848 --- [ main] c.c.mingyue.event.service.OrderService : 任務(wù)全部完成,總耗時:(6008)毫秒

          5.開啟異步


          啟動類增加 @EnableAsync 注解
          @EnableAsync@SpringBootApplicationpublic class MingYueSpringbootEventApplication {
          public static void main(String[] args) { SpringApplication.run(MingYueSpringbootEventApplication.class, args); }}

          Listener 類需要開啟異步的方法增加 @Async 注解
          @Async@SneakyThrows@EventListener(MsgEvent.class)public void sendMsg(MsgEvent event) { String orderId = event.getOrderId(); long start = System.currentTimeMillis(); log.info("開發(fā)發(fā)送短信"); log.info("開發(fā)發(fā)送郵件"); Thread.sleep(4000); long end = System.currentTimeMillis(); log.info("{}:發(fā)送短信、郵件耗時:({})毫秒", orderId, (end - start));}

          6.單測執(zhí)行(異步)

          發(fā)送短信的線程顯示 task-1,主線程結(jié)束后(總耗時:(2017)毫秒)控制臺停止打印了
          2022-04-24 10:30:59.002 INFO 59448 --- [ main] c.c.m.e.listener.OrderProductListener : 732171109:校驗(yàn)訂單商品價格耗時:(2009)毫秒2022-04-24 10:30:59.009 INFO 59448 --- [ main] c.c.mingyue.event.service.OrderService : 任務(wù)全部完成,總耗時:(2017)毫秒2022-04-24 10:30:59.028 INFO 59448 --- [ task-1] c.c.mingyue.event.listener.MsgListener : 開發(fā)發(fā)送短信2022-04-24 10:30:59.028 INFO 59448 --- [ task-1] c.c.mingyue.event.listener.MsgListener : 開發(fā)發(fā)
          <END>

          推薦閱讀:

          7行代碼讓B站崩潰3小時,竟因“一個詭計多端的0”

          Spring Security 實(shí)現(xiàn)動態(tài)權(quán)限菜單方案(附源碼)

          互聯(lián)網(wǎng)初中高級大廠面試題(9個G)

          內(nèi)容包含Java基礎(chǔ)、JavaWeb、MySQL性能優(yōu)化、JVM、鎖、百萬并發(fā)、消息隊(duì)列、高性能緩存、反射、Spring全家桶原理、微服務(wù)、Zookeeper......等技術(shù)棧!

          ?戳閱讀原文領(lǐng)取!                                  朕已閱 

          瀏覽 122
          點(diǎn)贊
          評論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報
          評論
          圖片
          表情
          推薦
          點(diǎn)贊
          評論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  九九热视频精品在线 | 亚洲色导航五月 | 草久97在线视频免费观 | 亚洲男女激情 | 天天摸天天操天天日 |