前言
当我们将对象交给spring管理的时候,经常会做一些初始化操作,初始化有很多种实现方法:static静态代码块、init()方法、构造方法(不建议)等等方式,但是有时我们会引用依赖,而依赖的加载时机经常会造成困扰绕,下面就一次性梳理明白,而且我们还可以通过一些手段改变依赖的加载时机。
代码测试
直接上代码:
@Component
public class Test {
@Autowired
private ApplicationContext context;
static {
System.out.println("执行静态方法");
}
public Test() {
System.out.println("执行无参构造方法");
System.out.println("无参构造方法中context=" + context);
}
@PostConstruct
public void init() {
System.out.println("init初始化方法执行");
System.out.println("init初始化方法中context=" + context);
}
}
项目启动后,执行结果如下:
执行静态方法
执行无参构造方法
无参构造方法中context=null
init初始化方法执行
init初始化方法中context=org.springframework.boot.web.servlet.context.AnnotationConfigServletWebServerApplicationContext@21362712, started on Fri Sep 18 18:22:33 CST 2020, parent: org.springframework.context.annotation.AnnotationConfigApplicationContext@5f0e9815
由结果可知,Spring启动时,加载顺序:
类的属性---->static静态方法---->无参构造方法---->注入@Autowired修饰的依赖---->执行@PostConstruct修饰的init()方法。
好了,清楚了上面初始化与依赖的加载时机,我们还可以改变依赖的加载顺序,将上面的类改变如下:
@Component
public class Test {
static {
System.out.println("执行静态方法");
}
@Autowired
public Test(ApplicationContext context) {
System.out.println("执行无参构造方法");
System.out.println("无参构造方法中context=" + context);
}
@PostConstruct
public void init() {
System.out.println("init初始化方法执行");
}
}
执行结果,可以看到,在构造方法执行的时候,依赖已经注入了,这与上面的结果是完全不同的。
执行静态方法
执行无参构造方法
无参构造方法中context=org.springframework.boot.web.servlet.context.AnnotationConfigServletWebServerApplicationContext@23a9ba52, started on Fri Sep 18 18:26:06 CST 2020, parent: org.springframework.context.annotation.AnnotationConfigApplicationContext@5812f68b
init初始化方法执行
可能会有人有疑问:spring会默认加载Test的无参构造方法,注意,如果类中写了有参构造,会默认覆盖掉无参构造,所以,如果写有参构造,必须显式的写出无参构造,否则会操作的。为什么Test类的构造方法会有一个参数ApplicationContext context,这还是无参构造吗?答案:是。
将@Autowired用在无参构造方法上,然后将要注入的ApplicationContext context作为参数。这样在执行构造方法之前就会将context注入进来,只是改变了依赖的注入时机,可以和上面的结构对比一下,就会明白的。