spring为何推荐使用构造器注入

在idea中使用field注入时,提示

Field injection is not recommended.

Inspection info: Spring Team recommends: "Always use constructor based dependency injection in your beans. Always use assertions for mandatory dependencies"

spring团队为何推荐使用构造器注入,而不推荐方法注入呢?
下面将对常见的三种构造器注入方式进行介绍。

filed注入

使用方法

@Controllerpublic classFooController{
    @Autowired  // @Inject  
    private FooService fooService;
    //简单的使用例子,下同  
    publicListlistFoo(){
      return fooService.list();
    }
}

该种注入方式是使用开发中最常见的注入方式。

优点

  1. 注入方式简单:加入要注入的字段,附上注解@AutoWired
  2. 整体代码简洁明了

缺点

  1. 对于IOC容器以外的环境,除了使用反射来提供它需要的依赖之外,无法复用该实现类。而且将一直是个潜在的隐患,因为你不调用将一直无法发现NullPointException的存在
  2. 使用field注入可能会导致循环依赖
public class A {
    @Autowired
    private B b;
}

public class B {
    @Autowired
    private A a;
}

constructor注入

使用方法

@Controller
public class FooController {  
    private final FooService fooService;
  
    @Autowired
    public FooController(FooService fooService) {
        this.fooService = fooService;
    }
}

优点

  • 依赖不可变 components as immutable objects ,即注入对象为final
  • 依赖不可为空required dependencies are not null,省去对注入参数的检查。当要实例化FooController的时候,由于只有带参数的构造函数,spring注入时需要传入所需的参数,所以有两种情况:1) 有该类型的参数传入 => ok; 2) 无该类型参数传入,报错
  • 提升了代码的可复用性:非IOC容器环境可使用new实例化该类的对象。
  • 避免循环依赖:如果使用构造器注入,在spring项目启动的时候,就会抛出:BeanCurrentlyInCreationException:Requested bean is currently in creation: Is there an unresolvable circular reference?从而提醒你避免循环依赖,如果是field注入的话,启动的时候不会报错,在使用那个bean的时候才会报错。

缺点

  • 当注入参数较多时,代码臃肿。

setter注入

使用方法

@Controller
public class FooController {
    private FooService fooService;

    @Autowired
    public void setFooService(FooService fooService) {
        this.fooService = fooService;
    }
}

优点

  1. 相比构造器注入,当注入参数太多或存在非必须注入的参数时,不会显得太笨重
  2. 允许在类构造完成后重新注入

缺点

答疑

Q1:如果有大量的依赖要注入,构造方法注入不会显得很臃肿吗?

对于这个问题,说明你的类当中有太多的责任,那么你要好好想一想是不是自己违反了类的单一性职责原则,从而导致有这么多的依赖要注入。

Q2:是不是其他的注入方式都不适合用了呢?

当然不是,存在即是合理!setter的方式既然一开始被Spring推荐肯定是有它的道理,像之前提到的setter的方式能用让类在之后重新配置或者重新注入,就是其优点之一。除此之外,如果一个依赖有多种实现方式,我们可以使用@Qualifier,在构造方法里选择对应的名字注入,也可以使用field或者setter的方式来手动配置要注入的实现。

转载: http://www.cnblogs.com/joemsu/p/7688307.html

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 文章作者:Tyan博客:noahsnail.com 3.4 Dependencies A typical ente...
    SnailTyan阅读 4,228评论 2 7
  • 2.1 我们的理念是:让别人为你服务 IoC是随着近年来轻量级容器(Lightweight Container)的...
    好好学习Sun阅读 2,746评论 0 11
  • 我想说,在未来的时代,如果你只掌握一门技术,一项本领,那么,在时代的不断创新与发展的潮流面前,你肯定会觉得力不从心...
    端儿阅读 81评论 0 0
  • 一、点击div时页面进行跳转,通过onclick事件来实现 1.通过window.open函数 在新窗口跳转 在...
    量子波阅读 230评论 0 0
  • 那时的他时常在想:他和她,贺佑树和许柠柠是怎么走到了如今这一步? 曾经好得盖同一张被子,如今只有见面时敷衍疏离的礼...
    凌树里阅读 476评论 0 4