大公司为什么禁止在SpringBoot项目中使用@Autowired注解?

Spring官方已不推荐使用Autowired字段/属性注入bean,,一些大公司的新项目也明令禁止使用了 。
1. 说明
最近公司升级框架,由原来的spring framerwork 3.0升级到5.0,然后写代码的时候突然发现idea在属性注入的@Autowired注解上给出警告提示,就像下面这样的,也挺懵逼的,毕竟这么写也很多年了 。
Field injection is not recommended
查阅了相关文档了解了一下,原来这个提示是spring framerwork 4.0以后开始出现的,spring 4.0开始就不推荐使用属性注入,改为推荐构造器注入和setter注入 。
下面将展示了spring框架可以使用的不同类型的依赖注入,以及每种依赖注入的适用情况 。
2. 依赖注入的类型
尽管针对spring framerwork 5.1.3的文档只定义了两种主要的依赖注入类型,但实际上有三种:
 

  • 基于构造函数的依赖注入
  • 基于setter的依赖注入
  • 基于字段的依赖注入
 
其中基于字段的依赖注入被广泛使用,但是idea或者其他静态代码分析工具会给出提示信息,不推荐使用 。
甚至可以在一些Spring官方指南中看到这种注入方法:
大公司为什么禁止在SpringBoot项目中使用@Autowired注解?

文章插图
 
2.1 基于构造函数的依赖注入
在基于构造函数的依赖注入中,类构造函数被标注为@Autowired,并包含了许多与要注入的对象相关的参数 。
@Component public class ConstructorBasedInjection { private final InjectedBean injectedBean; @Autowired public ConstructorBasedInjection(InjectedBean injectedBean) { this.injectedBean = injectedBean; } }
然后在spring官方文档中,@Autowired注解也是可以省去的 。
public class SimpleMovieLister { // the SimpleMovieLister has a dependency on a MovieFinder private MovieFinder movieFinder; // a constructor so that the Spring container can inject a MovieFinder public SimpleMovieLister(MovieFinder movieFinder) { this.movieFinder = movieFinder; } // business logic that actually uses the injected MovieFinder is omitted... }
【大公司为什么禁止在SpringBoot项目中使用@Autowired注解?】基于构造函数注入的主要优点是可以将需要注入的字段声明为final,使得它们会在类实例化期间被初始化,这对于所需的依赖项很方便 。
2.2 基于Setter的依赖注入
在基于setter的依赖注入中,setter方法被标注为@Autowired 。一旦使用无参数构造函数或无参数静态工厂方法实例化Bean,为了注入Bean的依赖项,Spring容器将调用这些setter方法 。
@Component public class SetterBasedInjection { private InjectedBean injectedBean; @Autowired public void setInjectedBean(InjectedBean injectedBean) { this.injectedBean = injectedBean; } }
和基于构造器的依赖注入一样,在官方文档中,基于Setter的依赖注入中的@Autowired也可以省去 。
public class SimpleMovieLister { // the SimpleMovieLister has a dependency on the MovieFinder private MovieFinder movieFinder; // a setter method so that the Spring container can inject a MovieFinder public void setMovieFinder(MovieFinder movieFinder) { this.movieFinder = movieFinder; } // business logic that actually uses the injected MovieFinder is omitted... } 2.3 基于属性的依赖注入
在基于属性的依赖注入中,字段/属性被标注为@Autowired 。一旦类被实例化,Spring容器将设置这些字段 。
@Component public class FieldBasedInjection { @Autowired private InjectedBean injectedBean; }
正如所看到的,这是依赖注入最干净的方法,因为它避免了添加样板代码,并且不需要声明类的构造函数 。代码看起来很干净简洁,但是正如代码检查器已经向我们暗示的那样,这种方法有一些缺点 。
3. 基于字段的依赖注入缺陷 3.1 不允许声明不可变域
基于字段的依赖注入在声明为 final / immutable 的字段上不起作用,因为这些字段必须在类实例化时实例化 。声明不可变依赖项的唯一方法是使用基于构造器的依赖注入 。
3.2 容易违反单一职责设计原则
在面向对象的编程中,五大设计原则SOLID被广泛应用,(国内一般为六大设计原则),用以提高代码的重用性,可读性,可靠性和可维护性
S在SOLID中代表单一职责原则,即即一个类应该只负责一项职责,这个类提供的所有服务都应该只为它负责的职责服务 。
使用基于字段的依赖注入,高频使用的类随着时间的推移,我们会在类中逐渐添加越来越多的依赖项,我们用着很爽,很容易忽略类中的依赖已经太多了 。但是如果使用基于构造函数的依赖注入,随着越来越多的依赖项被添加到类中,构造函数会变得越来越大,我们一眼就可以察觉到哪里不对劲 。


推荐阅读