亲宝软件园·资讯

展开

springboot2启动时执行,初始化(或定时任务)servletContext问题

亦寒2017 人气:0

springboot2启动时执行,初始化(或定时任务)servletContext

需求:springboot 启动后自动执行,初始化数据,并将数据放到 servletContext 中。

首先,不可使用 ServletContextListener 即不能用 @WebListener ,因为 servlet 容器初始化后,spring 并未初始化完毕,不能使用 @Autowired 注入 spring 的对象。

代码如下:

可以实现 ApplicationListener

@Component
public class SettingDataInitListener implements ApplicationListener<ContextRefreshedEvent> {
    @Override
    public void onApplicationEvent(ContextRefreshedEvent contextRefreshedEvent) {
        WebApplicationContext webApplicationContext = 
            (WebApplicationContext)contextRefreshedEvent.getApplicationContext();
        ServletContext servletContext = webApplicationContext.getServletContext();
        servletContext.setAttribute("key", "value");
    }
}

使用注解注入

service类里注入 servletContext

@Autowired private ServletContext servletContext;

service类里要启动执行的方法加上注解

@PostConstruct

在定时任务里,也可以注入 servletContext,进行定时操作

@Component
public class MyTask {
    @Autowired
    private ServletContext servletContext;

    // 秒 分 时 日 月 周
    @Scheduled(cron = "0 * * * * *")
    public void resetDays() {
        // servletContext
    }
}

springboot启动时初始化数据的几种方式

在我们用springboot搭建项目的时候,经常碰到在项目启动时初始化一些字典数据、地市数据、等各类需求,针对这种需求Spring与Spring boot为我们提供了以下几种方案供我们选择:

InitializingBean、BeanPostProcessor接口

Spring的事件机制: 实现 ApplicationListener 接口

一、ApplicationRunner与CommandLineRunner

如果需要在SpringApplication启动时执行一些特殊的代码,可以通过实现ApplicationRunner或CommandLineRunner接口,这两个接口都提供单一的run方法,且run方法仅在SpringApplication.run(…)完成之前调用。

区别:参数不一样,CommandLineRunner的参数是最原始的参数,没有进行任何处理,ApplicationRunner的参数是ApplicationArguments

ApplicationRunner接口只需要自己创建类实现ApplicationRunner接口

/**
 * @author 重庆阿汤哥
 * @Description: 测试
 * @date 2021/11/26  
 */
@Component
@Slf4j
public class ApplicationRunnerTest implements ApplicationRunner {

    @Override
    public void run(ApplicationArguments args) throws Exception {
        log.info("ApplicationRunner init data.....");
    }
}

CommandLineRunner

对于这个接口而言,我们可以通过Order注解或者使用Ordered接口来指定调用顺序,@Order()中的值越小,优先级越高

/**
 * @author 重庆阿汤哥
 * @Description: 测试
 * @date 2021/11/26  
 */
@Component
@Slf4j
@Order(1)
public class CommandLineRunnerTest implements CommandLineRunner {

    @Override
    public void run(String... args) throws Exception {
        log.info("CommandLineRunner init data.....");
    }
}

二、Spring Bean初始化的InitializingBean,init-method和PostConstruct

InitializingBean接口、BeanPostProcessor接口

InitializingBean接口为bean提供了初始化方法的方式,它只包括afterPropertiesSet()方法。

在spring初始化bean的时候,如果bean实现了InitializingBean接口,在对象的所有属性被初始化后之后才会调用afterPropertiesSet()方法

/**
 * @author 重庆阿汤哥
 * @Description: 测试
 * @date 2021/11/26  
 */
@Component
@Slf4j

@Component
public class InitialingzingBeanTest implements InitializingBean {

    @Override
    public void afterPropertiesSet() throws Exception {
       log.info("InitializingBean init....");
    }
}

@PostConstruct

/**
 * @author 重庆阿汤哥
 * @Description: 测试
 * @date 2021/11/26  
 */
@Component
@Slf4j
public class DynamicRouteMonitor {
   
    @PostConstruct
    public void init() {
        log.info("gateway route init...");
        }
}

BeanPostProcessor接口

可以用于判断某些特定类加载完成后才能初始化数据的场景,只需要自己实现该接口中的方法进行前置条件判断

public interface BeanPostProcessor {
    @Nullable
    default Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
        return bean;
    }

    @Nullable
    default Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        return bean;
    }
}

三、Spring的事件机制

在Spring中,默认对ApplicationEvent事件提供了如下支持:

利用ContextRefreshedEvent事件进行初始化操作

/**
 * @author 重庆阿汤哥
 * @Description:容器初始化完整后初始字典数据
 * @date 2021/11/26  10:51
 */
@Component
public class ApplicationStartup implements ApplicationListener<ContextRefreshedEvent> {
    public void onApplicationEvent(ContextRefreshedEvent event) {
        //在容器加载完毕后获取dao层来操作数据库
        ISysDictTypeService sysDictTypeService = (ISysDictTypeService) event.getApplicationContext().getBean(ISysDictTypeService.class);

        sysDictTypeService.initDict();

        IProvCityService provCityService = (IProvCityService) event.getApplicationContext().getBean(IProvCityService.class);
        provCityService.initProvCity();

    }

    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
}

这几种方式都可以满足我们日常开发的需求,针对具体场景使用对应的方案,在微服务应用中使用也较广泛。

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持。

加载全部内容

相关教程
猜你喜欢
用户评论