likes
comments
collection
share

Spring选择哪种注入方式

作者站长头像
站长
· 阅读数 77

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注入尽管代码上简洁有效,但是会带来一些问题:

  1. 对于IOC容器以外的环境,除了使用反射来提供它需要的依赖之外,无法复用该实现类
  2. 不调用依赖bean时,不会发现空指针的异常,在运行时调用会报异常
  3. 使用field注入可能会导致循环依赖,即A里面注入B,B里面又注入A
  4. 会造成依赖臃肿,职责过多,使用Filed注入时,添加数量不受限制(没有警告),可能会注入过多的依赖项,违反了单一职责原则。

Setter方法注入是Spring 3.x版本推荐的注入方式,因为setter的方式能用让类在之后重新配置或者重新注入,但是

  • 写起来比较麻烦,每个依赖都需要一个set方法
  • 可以重新配置或注入虽然方便,但是有些依赖需要不可变,使用此种方式产生不确定性

构造器注入是Spring 4.x 版本推荐的注入方式,主要优点有:

  1. 依赖不可变,使用final定义,保证注入后的不可变
  2. 依赖不为null,使用构造函数传参的方式,在实例化对象传参时如果参数为null,则报错,提前避免了空指针异常
  3. 在传参时传入的是依赖对象,而该对象传入时会保证对象类已经完成了初始化,也同时保证了对象不为null
  4. 如果构造器注入时发生了循环依赖,在项目启动时就会报错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
评论
请登录