有时候我们需要在应用启动完成后执行一些特定的操作,比如:
删除一些临时文件或者Redis中的缓存
将一些字典类的数据加载到缓存,这样就不用每次去数据库中查了,有些关联数据从缓存中取得赋值就可以了,不再需要联接查询
启动一个计时器
这在没有使用Spring Boot之前,在Spring框架下是通过ApplicationListener监听器来实现的,在Spring Boot下也有一些办法,可以通过使用CommandLineRunner和ApplicationRunner接口来实现。
在这两个接口中都有run()方法,在Spring Boot程序启动后会检测程序中是否有CommandLineRunner和ApplicationRunner接口的实例,如果存在,则会执行对应实现类中的run()方法,而且只执行一次。
Spring Boot程序启动过程中会实例化ApplicationContext,然后创建所有的Bean。
如果存在多个CommandLineRunner和ApplicationRunner接口的实例,那么它们的执行顺序可以通过@Order注解和Ordered接口来标识,数字越小,优先级越高。
看如下实例:
TaskRunner类:
public class TaskRunner implements ApplicationRunner, Ordered {
private static final Logger logger = LoggerFactory.getLogger(TaskRunner.class);
@Override
public int getOrder() {
return 2;
}
@Override
public void run(ApplicationArguments args) throws Exception {
logger.info("task runner");
}
}
StartupRunner类:
@Order(1)
public class StartupRunner implements CommandLineRunner {
private static final Logger logger = LoggerFactory.getLogger(StartupRunner.class);
@Autowired
@Qualifier("primaryDataSource")
private DataSource primaryDataSource;
@Autowired
@Qualifier("secondaryDataSource")
private DataSource secondDataSource;
@Override
public void run(String... args) throws Exception {
logger.info("startup runner");
System.out.print(primaryDataSource.getConnection().getMetaData().getURL());
System.out.print(secondDataSource.getConnection().getMetaData().getURL());
}
}
然后在Application主类中定义这两个bean,由于@SpringBootApplication注解中包含@Configuration注解,所以@Bean注解在其中也会起作用,如下:
@SpringBootApplication
public class SpringBootTemplateApplication {
public static void main(String[] args) {
SpringApplication.run(SpringBootTemplateApplication.class, args);
}
@Bean
public StartupRunner startupRunner(){
return new StartupRunner();
}
@Bean
public TaskRunner taskRunner(){
return new TaskRunner();
}
}