实际业务开发过程中,模式业务逻辑可能非常复杂,轻松核心业务 + N个子业务。实现如果都放到一块儿去做,业务代码可能会很长,解耦耦合度不断攀升,模式维护起来也麻烦,轻松甚至头疼。实现还有一些业务场景不需要在一次请求中同步完成,业务比如邮件发送、解耦短信发送等。模式 MQ 确实可以解决这个问题,轻松但 MQ 重啊,实现非必要不提升架构复杂度。业务 针对这些问题,解耦我们了解一下 Spring Event。亿华云计算 Spring Event(Application Event)其实就是一个观察者设计模式,一个 Bean 处理完成任务后希望通知其它 Bean 或者说一个 Bean 想观察监听另一个Bean 的行为。 Spring Event 用来解耦业务真的贼好用! 定义事件,继承 ApplicationEvent 的类成为一个事件类; / * @author Strive * @date 2022/4/22 18:00 * @description */ @Data @ToString public class OrderProductEvent extends ApplicationEvent { /** 该类型事件携带的信息 */ private String orderId; public OrderProductEvent(Object source, String orderId) { super(source); this.orderId = orderId; } 监听并处理事件,实现 ApplicationListener 接口或者使用 @EventListener 注解; / * 实现 ApplicationListener 接口,并指定监听的事件类型 * @author Strive * @date 2022/4/24 09:09 * @description */ @Slf4j @Component public class OrderProductListener implements ApplicationListener /** 使用 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)); } 发布事件,通过 ApplicationEventPublisher 发布事件; 推荐个人知识总结网站:www.java-family.cn。 / * @author Strive * @date 2022/4/24 09:25 * @description */ @Slf4j @Service @RequiredArgsConstructor public 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.单测执行@SpringBootTest public 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)毫秒 有些业务场景不需要在一次请求中同步完成,比如邮件发送、短信发送等。 1. 自定义事件@Data @AllArgsConstructor public class MsgEvent { /** 该类型事件携带的信息 */ public String orderId; 推荐使用 @EventListener 注解; @Slf4j @Component public 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. 单测执行(同步)@Test public 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)毫秒 启动类增加 @EnableAsync 注解; @EnableAsync @SpringBootApplication public 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)); 发送短信的线程显示 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 : 开发发送短信Spring Event 同步使用
Spring Event 异步使用