Spring选择哪种注入方式
1. Spring常见的注入方式
1.1 Field注入
field即在变量上直接使用注解进行注入,内部使用反射的方式实现注入到field中。
- 优点:代码简洁有效,最为开发人员喜欢使用
- 缺点:依赖对象可能为null而报空指针异常,容易出现循环依赖问题
@Autowired
private GoodsCategoryService goodsCategoryService;
1.2 setter注入
spring 3.x版本中推荐使用的注入方式,通过在set方法上使用注解完成对象注入。
- 优点:选择性注入,可有可无,依赖不会影响整个项目运行
- 缺点:在对象实例化后,依赖信息依然可以通过set方法修改,不太好
private GoodsCategoryService goodsCategoryService;
@Autowired
public setGoodsCategoryService(GoodsCategoryService goodsCategoryService){
this.goodsCategoryService = goodsCategoryService;
}
1.3 构造器注入
spring 4.x版本时Spring团队推荐使用的注入方式,将注解标注在构造函数上,对象以参数的形式传递,在构造函数中完成注入对象的初始化。
- 优点:变量定义使用强制性的显式注入(final关键字保证不可变)、避免空指针和循环依赖,
- 缺点:在需要注入的依赖较多时,使用构造方法臃肿
private final GoodsCategoryService goodsCategoryService;
@Autowired
public GoodServiceImpl(GoodsCategoryService goodsCategoryService){
this.goodsCategoryService = goodsCategoryService;
}
2. Spring为什么推荐使用构造器注入
Field注入尽管代码上简洁有效,但是会带来一些问题:
- 对于IOC容器以外的环境,除了使用反射来提供它需要的依赖之外,无法复用该实现类
- 不调用依赖bean时,不会发现空指针的异常,在运行时调用会报异常
- 使用field注入可能会导致循环依赖,即A里面注入B,B里面又注入A
- 会造成依赖臃肿,职责过多,使用Filed注入时,添加数量不受限制(没有警告),可能会注入过多的依赖项,违反了单一职责原则。
Setter方法注入是Spring 3.x版本推荐的注入方式,因为setter的方式能用让类在之后重新配置或者重新注入,但是
- 写起来比较麻烦,每个依赖都需要一个set方法
- 可以重新配置或注入虽然方便,但是有些依赖需要不可变,使用此种方式产生不确定性
构造器注入是Spring 4.x 版本推荐的注入方式,主要优点有:
- 依赖不可变,使用final定义,保证注入后的不可变
- 依赖不为null,使用构造函数传参的方式,在实例化对象传参时如果参数为null,则报错,提前避免了空指针异常
- 在传参时传入的是依赖对象,而该对象传入时会保证对象类已经完成了初始化,也同时保证了对象不为null
- 如果构造器注入时发生了循环依赖,在项目启动时就会报错BeanCurrentlyInCreationException,而Field注入只有在使用时才会报错
3. @Autowired、@Resource和@Inject的区别
3.1 @Autowired
- @Autowired注解是Spring2.5之后带入的注解,通过AutowiredAnnotationBeanPostProcessor类实现注入
- @Autowired可以标注在CONSTRUCTOR、METHOD、PARAMETER、FIELD、ANNOTATION_TYPE之上
- @Autowired默认根据bean类型进行自动装配,byType
- @Autowired注入时如果发现同类型有多个bean,此时需要使用@Qualifier注解来指定bean的name,即byName
- @Autowired注解有属性required,默认为true,设置为false则表示未找到对应bean时不抛出异常
3.2 @Resource
- @Resource时javax.annotation包下的注解
- @Resource可以标注在TYPE, FIELD, METHOD之上
- @Resource是根据属性名称进行自动装配的,其拥有属性name,可以根据属性值指定装配bean的name
3.3 @Inject
@Inject注解需要导入javax.inject.Inject包,能实现注入 @Inject可以标注在CONSTRUCTOR、METHOD、FIELD之上 @Inject注解是根据类型自动装配的,如果需要指定名称,则需要配合@Named注解
4. 总结
尽管Spring 4.x 推荐使用构造器注入,但是我们仍然可以使用其他方式,且可以使用混合方式注入
- 不可变的注入使用构造器
- 可变可选注入使用setter方法
- 对于多实现的依赖注入时,可以使用Filed注入或者setter方法来指定注入的实现
- 实际使用Field注入的仍然很多,大部分是一些遗留问题,后续新项目可以尝试尽量避免Field注入
转载自:https://juejin.cn/post/7021902992706109476