Spring ApplicationListener ContextRefreshedEvent 多次执行问题及源码分析

我们可能会使用到如下方法,来在Spring初始化完成后,执行某些程序:

@Component

public class SpringContextRefreshedListener implements 

                ApplicationListener<ContextRefreshedEvent> {

 

    @Override

    public void onApplicationEvent(ContextRefreshedEvent event) {

        System.out.println("ContextRefreshedListener execute......");

        System.out.println(event.toString());

        System.out.println(event.getTimestamp());

        ApplicationContext application = event.getApplicationContext();

        ..... do something 

    }

 

但是,这个方法可能会执行多次!!

 

具体来说, 每加载完一次context,就会执行一次ContextRefreshedEvent

 * 而且每次执行,都会再执行一次parent的ContextRefreshedEvent

 * 也就是说,如果web.xml里面配置了两个Context,

 * 且两个Context都配置了ContextRefreshedEvent Listener,那么ContextRefreshedEvent会执行3次!

 * 而且更郁闷的是,后一个Context执行的两次ContextRefreshedEvent是一样的。

 

 具体源码分析:AbstractApplicationContext.publishEvent

if (this.parent != null) {

    if (this.parent instanceof AbstractApplicationContext) {

        ((AbstractApplicationContext) this.parent).publishEvent(event, eventType);

    }

    else {

        this.parent.publishEvent(event);

    }

}

parent的publishEvent,跟儿子的是一样的,所以两次事件效果一模一样。

 

网上很多人都遇到这个问题,但是解决方法都是 if (this.parent == null) {},治标不治本。

 

为了避免这种情况,根本的解决方法,就是  ApplicationListener<ContextRefreshedEvent> 应该和 ApplicationContext 一对一。如果有两个 ApplicationContext ,且两个ApplicationContext 都配置了ApplicationListener<ContextRefreshedEvent>,那么就应该执行两次。

 

如果想 ApplicationListener<ContextRefreshedEvent> 只执行一次,那就只应该把它配置在 其中一个ApplicationContext 中,另外一个ApplicationContext 不配置也就是说,使ApplicationListener<ContextRefreshedEvent>的实现类,只被其中一个ApplicationContext 加载到。

假如说是只给第二个ApplicationContext 配置了ApplicationListener<ContextRefreshedEvent>,名字叫MyApplicationListener,那么它会执行自己的ContextRefreshedEvent,此时MyApplicationListener被调用,也会调用parent,但是由于parent没有配置ApplicationListener<ContextRefreshedEvent>,所以不会有效果。那么最终就只有MyApplicationListener被调用了一次。这才是正确的解决方案。

 

参见另一篇文章:

Spring配置加载ContextLoaderListener和DispatcherServlet的区别和关系

 

  • 3
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
Spring框架中,我们可以通过实现ApplicationListener接口来监听Spring容器的事件,从而在特定事件发生时执行一些特定的处理。 例如,在项目启动时我们可能需要进行一些初始化操作,比如读取配置文件、连接数据库等等。此时,我们可以实现ApplicationListener接口,在onApplicationEvent方法中编写初始化逻辑。 下面是一个简单的示例代码: ```java @Component public class MyApplicationListener implements ApplicationListener<ContextRefreshedEvent> { @Override public void onApplicationEvent(ContextRefreshedEvent event) { // 在Spring容器初始化完成后执行 // 这里可以写一些项目启动时的初始化逻辑 System.out.println("项目启动完成,执行初始化操作..."); } } ``` 在上面的代码中,我们使用@Component注解将MyApplicationListener类纳入Spring容器的管理中,并实现了ApplicationListener接口,并指定了监听的事件类型为ContextRefreshedEvent,这是Spring容器初始化完成后会发布的事件。 当Spring容器初始化完成后,就会自动触发onApplicationEvent方法,执行我们在其中编写的初始化逻辑。 除了ContextRefreshedEvent事件,Spring容器还支持许多其他事件类型,例如ContextStartedEvent(容器启动时触发)、ContextStoppedEvent(容器停止时触发)等等。我们可以根据具体需求选择不同的事件类型来监听。 在底层实现上,Spring容器会通过调用ConfigurableApplicationContext.publishEvent()方法来发布事件,然后遍历所有实现了ApplicationListener接口的bean,并调用它们的onApplicationEvent()方法来处理相应的事件。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值