在项目中,经常有这样的需求,我们需要在项目启动完立即初始化一些数据(比如缓存等),以便后面调用使用。
1、CommandLineRunner
@Component
public class InitRunner implements CommandLineRunner{
@Override
public void run(String... args) throws Exception {
System.out.println("项目启动后立即执行。。。。。");
}
}
1、CommandLineRunner会在服务启动之后被立即执行。
2、CommandLineRunner可以有多个实现类,且多个直接可以用@Order注解确定执行顺序。
2、 @PostConstruct
@Component
public class Init {
@PostConstruct
private void init(){
System.out.println("PostConstruct注解初始化数据");
}
}
在项目还没有启动成功的时候,@PostConstruct已经执行完了,因为@PostConstruct是在Init类注入完成后立马执行的,它并不依赖于项目的启动。
一:@PostConstruct
不是spring提供的而是Java自己的注解。
Java中该注解的说明:@PostConstruct该注解被用来修饰一个非静态的void()方法。
被@PostConstruct修饰的方法会在服务器加载Servlet的时候运行,并且只会被服务器执行一次。PostConstruct注释用于需要依赖注入完成后才能执行任何初始化的方法上。 在类投入使用之前,必须调用此方法。
在类加载的时候,为当前类初始化一些数据,那么可以使用@PostConstruct注解。
【Servlet中增加了两个影响Servlet生命周期的注解,@PostConstruct和@PreDestroy,这两个注解被用来修饰一个非静态的返回值为void的方法。】
在使用spring框架时,在一个类内,如果有构造器(Constructor ),有@PostConstruct,还有@Autowired,他们的先后执行顺序为Constructor >> @Autowired >> @PostConstruct。
因为一个有声明注解的类文件(必须有声明,这样在项目初始化时候才会注入),在项目启动后,会对对象进行依赖注入,而初始化的动作会依赖于对象,所以假象上看,也类似于项目启动就会执行的操作,因此,我们也可以通过这样的形式,对数据进行初始化。
二:CommandLineRunner
spring boot可以通过CommandLineRunner接口实现启动加载功能,实现CommandLineRunner接口,然后重写run方法,在run方法内编写需要加载的内容。
我们可以写多个类继承CommandLineRunner接口,这些类之间,可以通过@Order(value=1)实现执行的先后顺序。【CommandLineRunner可以有多个,且多个直接可以用@Order(value=1)注解进行排序。】
CommandLineRunner会在服务启动之后被立即执行。
三:比较
@PostConstruct 要比实现CommandLineRunner的类加载的要早:CommandLineRunner 是在项目启动完之后加载。
@PostConstruct更针对性于当前类文件,而CommandLineRunner更服务于整个项目。