封装RocketMQ公共组件的时候,首先尝试使用Spring的IOC托管机制,后来发现配置文件读取、Spring实例工厂定义、bean实例初始化 等时间节点较难控制, 所以考虑使用工具类的静态方法中托管相关实例进行实现,但在项目启动的 ApplicationListener 中配置监听事件的时候,发现会被执行两次,采用如下的 方式可以解决这个问题。
问题原因
在web项目中如果同时集成了Spring和SpringMVC的话,上下文中会存在两个容器,即Spring的applicationContext.xml的父容器和 SpringMVC的applicationContext-mvc.xml的子容器。这两个容器有相同的生命周期,所以同一个事件,在不同容器启动过程中都会发送一次。 如此,明白了多出的一次事件是引入了SpringMVC造成的。基于第一节中的时间执行顺序,我们可以得出这样的结论:
root容器启动开始 –> 创建子容器mvc并启动 –> 子容器mvc启动完成 –> root容器继续启动 –> root容器启动完成
原方案
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.springframework.boot.context.event.ApplicationReadyEvent;
import org.springframework.context.ApplicationListener;
import org.springframework.stereotype.Component;
@Slf4j
@Component
@RequiredArgsConstructor
public class MqConsumeInit implements ApplicationListener<ApplicationReadyEvent> {
private final MqListenerDemo mqListenerDemo;
@Override
public void onApplicationEvent(ApplicationReadyEvent event) {
RocketMqUtil.consumer("platform-consumer", mqListenerDemo);
}
}
改正后的方案
通过比对 ApplicationContext 的实例类型来判断:
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.BeansException;
import org.springframework.boot.context.event.ApplicationReadyEvent;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.context.ApplicationListener;
import org.springframework.stereotype.Component;
@Slf4j
@Component
@RequiredArgsConstructor
public class MqConsumeInit implements ApplicationListener<ApplicationReadyEvent>, ApplicationContextAware {
private ApplicationContext applicationContext;
private final MqListenerDemo mqListenerDemo;
@Override
public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
this.applicationContext = applicationContext;
}
@Override
public void onApplicationEvent(ApplicationReadyEvent event) {
if (event.getApplicationContext().equals(this.applicationContext)) {
RocketMqUtil.consumer("platform-consumer", mqListenerDemo);
}
}
}
参考资料
- SpringBoot ApplicationListener 监听事件触发两次的问题
- Why Spring Boot Application logs that it started twice
- Spring事件处理——onApplicationEvent执行两次
- Spring事件发布与监听机制
开始写作吧
![image-alter](/image/post/2023/04/19/02/xxx.png)
文档信息
- 本文作者:Bob.Zhu
- 本文链接:https://adolphor.github.io/2023/04/19/02-spring-boot-applicationListener-execute-twice/
- 版权声明:自由转载-非商用-非衍生-保持署名(创意共享3.0许可证)