在使用@Bean注解的时候,发现@Autowired注入的对象一直是NULL,然后换一个类代码相同@Autowired注入的对象又不是NULL。苦恼了好几天,终于找到了玄机。
@Configuration
public class MyConfig {
@Autowired
private Table table;
@Autowired
private Fa fa;
@Bean
public Table table() {
Table table = new Table();
table.fa = fa;
System.out.println("Current table.fa is : " + table.fa);
return table;
}
public void print() {
System.out.println(table);
}
}
public class Table {
f
public Fa fa;
}
@Component
public class Fa {
@PostConstruct
void init() {
System.out.println("Fa init");
}
}
MyConfig
就是一个配置类,Table
就是一个普通的类包含了Fa
类。当你运行的时候你会发现fa
是null
:
对于这个结果一直很困惑,我查阅过很多代码不管是Spring官方,还是国内外各种Demo,都有在
@Bean
的方法中直接引用@Autowired
注入的对象,为什么我这里就不行呢?后来无意中发现@Authowired
的声明顺序就是对象的创建顺序。由于我的声明顺序如下:
@Autowired
private Table table;
@Autowired
private Fa fa;
Spring会先调用table()
方法,然而这个时候fa
还没有创建,所以输出是NULL
。
@Bean
public Table table() {
Table table = new Table();
table.fa = fa;
System.out.println("Current table.fa is : " + table.fa);
return table;
}
当我将Table
和Fa
的声明顺序调换后,输入如下:
为了进一步验证这个注入顺序,我编写了几个测试类:
@Component
public class Fa {
@PostConstruct
void init() {
System.out.println("Fa init");
}
}
@Component
public class Fb {
@PostConstruct
void init() {
System.out.println("Fb init");
}
}
@Component
public class Fc {
@PostConstruct
void init() {
System.out.println("Fc init");
}
}
@Component
public class Container {
@Autowired
private Fa fa;
@Autowired
private Fb fb;
@Autowired
private Fc fc;
}
我在Container
类中分别以不同的顺序声明Fa
、Fb
、Fc
,测试结果如下:
- 1
@Autowired
private Fa fa;
@Autowired
private Fb fb;
@Autowired
private Fc fc;
fa fb fc.png
- 2
@Autowired
private Fb fb;
@Autowired
private Fc fc;
@Autowired
private Fa fa;
fb fc fa.png
- 3
@Autowired
private Fc fc;
@Autowired
private Fa fa;
@Autowired
private Fb fb;
fc fa fb.png
从上面3轮的测试情况来看,Spring
@Autowired
的注入顺序与其声明顺序相关。
另外值得一提的是,Spring官方文档关于@Bean
依赖关系的声明提供了两种方法。第一种通过参数引入,第二种直接调用另一个@Bean
声明的方法。依赖声明可以确保需要的对象都已创建,而@Autowired
只是对象注入并不能保证需要的时候对象已经创建,因此在使用@Bean
的时候应该使用依赖声明而不是@Autowired
。
@Bean
public Table table(Fa fa) {//添加参数,声明table的创建需要Fa
Table table = new Table();
table.fa = fa;
Table table2 = table2();//直接调用table2()方法声明依赖关系
return table;
}
@Bean
public Table table2() {
Table table = new Table();
return table;
}
最后,需要注意的是直接调用@Bean
方法来声明依赖关系只有在@Configuration
注解的类中有效,普通类中调用只是普通的方法调用。
网友评论