Spring ApplicationListener使用方法及问题

本文介绍如何使用SpringBoot中的ApplicationListener接口实现启动监听,详细解释了如何通过实现onApplicationEvent方法来执行容器初始化后的操作,并避免了在多容器环境下重复执行的问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

使用场景

在一些业务场景中,当容器初始化完成之后,需要处理一些操作,比如一些数据的加载、初始化缓存、特定任务的注册等等。这个时候我们就可以使用Spring提供的ApplicationListener来进行操作。

用法

本文以在Spring boot下的使用为例来进行说明。首先,需要实现ApplicationListener接口并实现onApplicationEvent方法。把需要处理的操作放在onApplicationEvent中进行处理:

package com.secbro.learn.context;

import org.springframework.context.ApplicationListener;
import org.springframework.context.event.ContextRefreshedEvent;
/**
 * Created by other  on 2017/5/12.
 */
public class ApplicationStartListener implements ApplicationListener<ContextRefreshedEvent>{
    @Override
    public void onApplicationEvent(ContextRefreshedEvent contextRefreshedEvent) {
        System.out.println("我的父容器为:" + contextRefreshedEvent.getApplicationContext().getParent());
        System.out.println("初始化时我被调用了。");
    }
}

然后,实例化ApplicationStartListener这个类,在Spring boot中通过一个配置类来进行实例化:

import com.secbro.learn.context.ApplicationStartListener;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

/**
 * Created by other  on 2017/5/12.
 */
@Configuration
public class ListenerConfig {

    @Bean
    public ApplicationStartListener applicationStartListener(){
        return new ApplicationStartListener();
    }
}

随后,启动Spring boot服务,打印出一下内容:

我的父容器为:null
初始化时我被调用了。

从打印的结果可以看出,ApplicationStartListener的onApplicationEvent方法在容器启动时已经被成功调用了。而此时初始化的容器为root容器。

二次调用问题

此处使用Spring boot来进行操作,没有出现二次调用的问题。在使用传统的application.xml和project-servlet.xml配置中会出现二次调用的问题。主要原因是初始化root容器之后,会初始化project-servlet.xml对应的子容器。我们需要的是只执行一遍即可。那么上面打印父容器的代码用来进行判断排除子容器即可。在业务处理之前添加如下判断:

if(contextRefreshedEvent.getApplicationContext().getParent() != null){
            return;
}

这样其他容器的初始化就会直接返回,而父容器(Parent为null的容器)启动时将会执行相应的业务操作。

关联知识

在spring中InitializingBean接口也提供了类似的功能,只不过它进行操作的时机是在所有bean都被实例化之后才进行调用。根据不同的业务场景和需求,可选择不同的方案来实现。

———————————————————————
以上是参考个别网友的写法
lora系统监测里面应用贴图
package com.example.demo;

import com.example.demo.socket.NIOClient;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationListener;
import org.springframework.context.event.ContextRefreshedEvent;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;

/**
 * 版本信息: Copyright (c)2018
 * 公司信息: 浙江**信息技术有限公司
 * 开发人员: xiaohuang email:331088675@qq.com
 * 创建日期: 2018/11/22 21:32
 * 修改历史:
 * 功能描述:项目启动完成后将上下文放入springUtil,开启线程进行客户端连接socket服务端
 * ———————————————————————————
 */
@Component
@Order(1)
public class ApplicationStartup implements ApplicationListener<ContextRefreshedEvent> {
    private static final Logger logger = LoggerFactory.getLogger(ApplicationStartup.class);
    public void onApplicationEvent(ContextRefreshedEvent event) {
        if (event.getApplicationContext().getParent() == null) {
            ApplicationContext ac = event.getApplicationContext();
            SpringContextUtil.setApplicationContext(ac);
            //只有root application context 没有父容器
            //start the executor
        }
        System.out.println("spring boot 容器加载完成》》》》》》》》》》》》》》》》》》》》》》》》》》》");
        //ambda形式 1.8Thread新特性
        new Thread(() -> {
            while (true) {
                logger.info("开始执行NIOClient连接socket服务操作!");
                NIOClient client = new NIOClient();
                try {
                    client.initClient("121.40.183.20", 19910);
                    client.listen();
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        }).start();
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值