Spring——事件机制

Spring中的事件机制是基于观察者模式实现的一种组件间通信机制,用于解耦不同组件,实现松耦合的消息传递,它允许一个组件发送事件,其他组件接收并处理事件,而无需直接依赖彼此。

事件机制的组成

Spring事件机制包含三个核心部分:

  • 事件:继承ApplicationEvent的对象,封装需要传递的数据。Spring4.2+可以省略,任意对象都可以作为事件。
  • 事件发布者:通过ApplicationEventPublisher接口发布事件,通常由spring管理的bean实现。
  • 事件监听器:监听并处理事件的组件,通过@EventListener注解或实现ApplicationListener接口定义。

示例

事件:

首先自定义一个事件类,去实现ApplicationEvent,该父类没有无参构造:

java 复制代码
public class MyEvent extends ApplicationEvent {
    
    public MyEvent(Object source) {
        super(source);
    }
}

发布者:

在业务代码内注入ApplicationEventPublisher,通过它去发布事件:

java 复制代码
@Autowired
    private ApplicationEventPublisher applicationEventPublisher;
    public void enent(){
        //业务代码
        //发送事件
        applicationEventPublisher.publishEvent(new MyEvent(this));
    }

监听器:

监听事件类型,两种方式都试一遍:

java 复制代码
@Component
public class MyLin implements ApplicationListener<MyEvent> {
    @Override
    public void onApplicationEvent(MyEvent event) {
        System.out.println("监听到事件");
    }
}
java 复制代码
@Component
public class MyLin2 {
    //注意方法的参数类型是监听的类型
    @EventListener
    public void onApplicationEvent(MyEvent event) {
        System.out.println("监听到事件2");
    }
}

接下来模拟异步的情况:

SimpleApplicationEventMulticaster

SimpleApplicationEventMulticaster是 Spring 框架中事件机制的核心组件 ,负责将发布的事件多播(multicast) 给所有订阅该事件的监听器,是事件从发布者传递到监听器的 "中介"。

它的主要职责是:

  1. 接收发布者发布的事件;
  2. 找到所有监听该事件类型的监听器;
  3. 将事件分发给这些监听器执行处理逻辑。

特性:

  • 事件发布后,它会直接发布在事件的线程中调研监听器的处理方法,发布者需等待所有的监听器完毕后才能继续,可以过setTaskExector(Executor)设置线程池,会将监听器的处理逻辑交到线程池执行,发布者无需等待。
  • 支持通过 addApplicationListener(ApplicationListener) 手动注册监听器;

配置好SimpleApplicationEventMulticaster后,通过ApplicationEventPublish发送事件后会提交到配置的线程池中执行。

java 复制代码
    @Bean
    public SimpleApplicationEventMulticaster applicationEventMulticaster(ThreadPoolExecutor executor){
        SimpleApplicationEventMulticaster multicaster = new SimpleApplicationEventMulticaster();
        multicaster.setTaskExecutor(executor);
        return multicaster;
    }

异步事件监听器

java 复制代码
@Component
public class AsyncEventListener {
    
    private static final Logger logger = LoggerFactory.getLogger(AsyncEventListener.class);
    
    /**
     * 异步处理事件 - 不会阻塞主线程
     */
    @Async("taskExecutor")  // 指定使用配置的线程池
    @EventListener
    public void handleAsyncEvent(MyEvent event) {
        logger.info("异步事件处理开始 - 事件: {}", event.getMessage());
        logger.info("当前线程: {}", Thread.currentThread().getName());
        
        try {
            // 模拟耗时操作
            Thread.sleep(3000);
            logger.info("异步事件处理完成: {}", event.getMessage());
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
            logger.error("异步事件处理被中断", e);
        }
    }
    
    /**
     * 同步处理事件 - 会阻塞主线程
     */
    @EventListener
    public void handleSyncEvent(MyEvent event) {
        logger.info("同步事件处理开始 - 事件: {}", event.getMessage());
        logger.info("当前线程: {}", Thread.currentThread().getName());
        
        try {
            // 模拟耗时操作
            Thread.sleep(2000);
            logger.info("同步事件处理完成: {}", event.getMessage());
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
            logger.error("同步事件处理被中断", e);
        }
    }
}
java 复制代码
@SpringBootTest
public class EventTest {
    
    @Autowired
    private BusinessService businessService;
    
    @Test
    public void testAsyncEvent() {
        System.out.println("测试开始 - 主线程: " + Thread.currentThread().getName());
        
        long startTime = System.currentTimeMillis();
        businessService.doBusiness();
        long endTime = System.currentTimeMillis();
        
        System.out.println("业务方法执行时间: " + (endTime - startTime) + "ms");
        
        // 给异步事件处理留出时间
        try {
            Thread.sleep(5000);
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        }
    }
}

Spring 内置事件

Spring 框架自带一些内置事件,用于通知容器生命周期的关键节点,常见的有:

事件类型 触发时机
ContextRefreshedEvent Spring 容器初始化完成(所有 Bean 加载完成)
ContextStartedEvent 容器启动时(调用start()方法)
ContextStoppedEvent 容器停止时(调用stop()方法)
ContextClosedEvent 容器关闭时(调用close()方法)
ApplicationFailedEvent 应用启动失败时(Spring Boot 中常用)
相关推荐
戴誉杰22 分钟前
idea 2025.2 重置试用30天,无限期使用
java·ide·intellij-idea
小坏讲微服务35 分钟前
Spring Cloud Alibaba 2025.0.0 整合 ELK 实现日志
运维·后端·elk·spring cloud·jenkins
IT_陈寒1 小时前
JavaScript性能优化:10个V8引擎隐藏技巧让你的代码快30%
前端·人工智能·后端
rannn_1111 小时前
【Javaweb学习|黑马笔记|Day5】Web后端基础|java操作数据库
数据库·后端·学习·javaweb
无限进步_1 小时前
C语言atoi函数实现详解:从基础到优化
c语言·开发语言·c++·git·后端·github·visual studio
q***78781 小时前
Spring学习——新建module模块
java·学习·spring
学历真的很重要1 小时前
PyTorch 零基础入门:从张量到 GPU 加速完全指南
人工智能·pytorch·后端·深度学习·语言模型·职场和发展
Python私教1 小时前
fasttushare 技术架构设计
后端
q***11651 小时前
在Nginx上配置并开启WebDAV服务的完整指南
java·运维·nginx
是店小二呀2 小时前
使用Rust构建一个完整的DeepSeekWeb聊天应用
开发语言·后端·rust