Spring@Autowired注入顺序

2020-08-05  本文已影响0人  随风摇摆水杉树

在使用@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类。当你运行的时候你会发现fanull

table.fa is null.png
对于这个结果一直很困惑,我查阅过很多代码不管是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;
}

当我将TableFa的声明顺序调换后,输入如下:

table.fa is instance.png
为了进一步验证这个注入顺序,我编写了几个测试类:
@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类中分别以不同的顺序声明FaFbFc,测试结果如下:

@Autowired
private Fa fa;
@Autowired
private Fb fb;
@Autowired
private Fc fc;
fa fb fc.png
@Autowired
private Fb fb;
@Autowired
private Fc fc;
@Autowired
private Fa fa;
fb fc fa.png
@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注解的类中有效,普通类中调用只是普通的方法调用。

上一篇 下一篇

猜你喜欢

热点阅读