理论业务开发过程中,业务逻辑可能非常复杂,外围业务 + N个子业务。如果都放到一块儿去做,代码可能会很长,耦合度一直攀升,保护起来也麻烦,甚至头疼。还有一些业务场景不须要在一次申请中同步实现,比方邮件发送、短信发送等。

MQ 的确能够解决这个问题,但 MQ 重啊,非必要不晋升架构复杂度。

针对这些问题,咱们理解一下 Spring Event。

Spring Event 同步应用

Spring Event(Application Event)其实就是一个观察者设计模式,一个 Bean 解决实现工作后心愿告诉其它 Bean 或者说一个 Bean 想察看监听另一个Bean 的行为。

Spring Event 用来解耦业务真的贼好用!

1. 自定义事件

定义事件,继承 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. 定义监听器

监听并处理事件,实现 ApplicationListener 接口或者应用 @EventListener 注解

/** * 实现 ApplicationListener 接口,并指定监听的事件类型 * * @author Strive * @date 2022/4/24 09:09 * @description */@Slf4j@Componentpublic class OrderProductListener implements ApplicationListener<OrderProductEvent> {  /** 应用 onApplicationEvent 办法对音讯进行接管解决 */  @SneakyThrows  @Override  public void onApplicationEvent(OrderProductEvent event) {    String orderId = event.getOrderId();    long start = System.currentTimeMillis();    Thread.sleep(2000);    long end = System.currentTimeMillis();    log.info("{}:校验订单商品价格耗时:({})毫秒", orderId, (end - start));  }}

3. 定义发布者

公布事件,通过 ApplicationEventPublisher 公布事件

/** * @author Strive * @date 2022/4/24 09:25 * @description */@Slf4j@Service@RequiredArgsConstructorpublic class OrderService {  /** 注入ApplicationContext用来公布事件 */  private final ApplicationContext applicationContext;  /**   * 下单   *   * @param orderId 订单ID   */  public String buyOrder(String orderId) {    long start = System.currentTimeMillis();    // 1.查问订单详情    // 2.测验订单价格 (同步解决)    applicationContext.publishEvent(new OrderProductEvent(this, orderId));    // 3.短信告诉(异步解决)    long end = System.currentTimeMillis();    log.info("工作全副实现,总耗时:({})毫秒", end - start);    return "购买胜利";  }}

4.单测执行

@SpringBootTestpublic class OrderServiceTest {  @Autowired private OrderService orderService;  @Test  public void buyOrderTest() {    orderService.buyOrder("732171109");  }}

执行后果如下:

2022-04-24 10:13:17.535  INFO 44272 --- [           main] c.c.m.e.listener.OrderProductListener    : 732171109:校验订单商品价格耗时:(2008)毫秒2022-04-24 10:13:17.536  INFO 44272 --- [           main] c.c.mingyue.event.service.OrderService   : 工作全副实现,总耗时:(2009)毫秒

Spring Event 异步应用

有些业务场景不须要在一次申请中同步实现,比方邮件发送、短信发送等。

1. 自定义事件

@Data@AllArgsConstructorpublic class MsgEvent {  /** 该类型事件携带的信息 */  public String orderId;}

2. 定义监听器

举荐应用 @EventListener 注解

@Slf4j@Componentpublic class MsgListener {  @SneakyThrows  @EventListener(MsgEvent.class)  public void sendMsg(MsgEvent event) {    String orderId = event.getOrderId();    long start = System.currentTimeMillis();    log.info("开发发送短信");    log.info("开发发送邮件");    Thread.sleep(4000);    long end = System.currentTimeMillis();    log.info("{}:发送短信、邮件耗时:({})毫秒", orderId, (end - start));  }}

3. 定义发布者

/**  * 下单  * @param orderId 订单ID  */public String buyOrder(String orderId) {    long start = System.currentTimeMillis();    // 1.查问订单详情    // 2.测验订单价格 (同步解决)    applicationContext.publishEvent(new OrderProductEvent(this, orderId));    // 3.短信告诉(异步解决)    applicationContext.publishEvent(new MsgEvent(orderId));    long end = System.currentTimeMillis();    log.info("工作全副实现,总耗时:({})毫秒", end - start);    return "购买胜利";}

4. 单测执行(同步)

@Testpublic void buyOrderTest() {    orderService.buyOrder("732171109");}

执行后果如下:

2022-04-24 10:24:13.905  INFO 54848 --- [           main] c.c.m.e.listener.OrderProductListener    : 732171109:校验订单商品价格耗时:(2004)毫秒2022-04-24 10:24:13.906  INFO 54848 --- [           main] c.c.mingyue.event.listener.MsgListener   : 开发发送短信2022-04-24 10:24:13.907  INFO 54848 --- [           main] c.c.mingyue.event.listener.MsgListener   : 开发发送邮件2022-04-24 10:24:17.908  INFO 54848 --- [           main] c.c.mingyue.event.listener.MsgListener   : 732171109:发送短信、邮件耗时:(4002)毫秒2022-04-24 10:24:17.908  INFO 54848 --- [           main] c.c.mingyue.event.service.OrderService   : 工作全副实现,总耗时:(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("开发发送短信");    log.info("开发发送邮件");    Thread.sleep(4000);    long end = System.currentTimeMillis();    log.info("{}:发送短信、邮件耗时:({})毫秒", orderId, (end - start));}

6.单测执行(异步)

发送短信的线程显示 task-1,主线程完结后(总耗时:(2017)毫秒)控制台进行打印了

2022-04-24 10:30:59.002  INFO 59448 --- [           main] c.c.m.e.listener.OrderProductListener    : 732171109:校验订单商品价格耗时:(2009)毫秒2022-04-24 10:30:59.009  INFO 59448 --- [           main] c.c.mingyue.event.service.OrderService   : 工作全副实现,总耗时:(2017)毫秒2022-04-24 10:30:59.028  INFO 59448 --- [         task-1] c.c.mingyue.event.listener.MsgListener   : 开发发送短信2022-04-24 10:30:59.028  INFO 59448 --- [         task-1] c.c.mingyue.event.listener.MsgListener   : 开发发送邮件

最初说一句(别白嫖,求关注)

陈某每一篇文章都是精心输入,曾经写了3个专栏,整顿成PDF,获取形式如下:

  1. 《Spring Cloud 进阶》PDF:关注公众号:【Java后端面试官】回复关键词 Spring Cloud 进阶 获取!
  2. 《Spring Boot 进阶》PDF:关注公众号:【Java后端面试官】回复关键词 Spring Boot进阶 获取!
  3. 《Mybatis 进阶》PDF:关注公众号:【Java后端面试官】回复关键词 Mybatis 进阶 获取!

如果这篇文章对你有所帮忙,或者有所启发的话,帮忙点赞在看转发珍藏,你的反对就是我坚持下去的最大能源!

本文由mdnice多平台公布