首页 > 其他分享 >【注解使用】使用@Autowired注解警告Field injection is not recommended

【注解使用】使用@Autowired注解警告Field injection is not recommended

时间:2023-05-25 10:33:52浏览次数:44  
标签:依赖 Autowired Spring Field 字段 使用 注解 注入 构造函数


本文目录

一、背景描述

二、出现原因

2.1 不允许声明不可变域

2.2 容易违反单一职责设计原则

2.3 与依赖注入容器紧密耦合

2.4 隐藏依赖关系

三、解决方案

3.1 使用@Resource

3.2 使用于构造函数的依赖注入

3.3 使用于Setter的依赖注入

3.4 个人总结

四、参考文档


Field injection is not recommended(不再推荐使用字段注入)

一、背景描述

最近升级了Idea工具,然后在写代码时,突然发现报波浪线,说明这个地方不严谨,于是就找解决办法。

【注解使用】使用@Autowired注解警告Field injection is not recommended_java

二、出现原因

查阅了相关文档了解了一下,原来这个提示是spring framerwork 4.0以后开始出现的,spring 4.0开始就不推荐使用属性注入,改为推荐构造器注入和setter注入。

由于此注入方式是三种依赖注入方式中的一种,它是基于字段的依赖注入,spring 4.0以后发现使用此注解有很多缺陷,例如:

2.1 不允许声明不可变域

基于字段的依赖注入在声明为final/immutable的字段上不起作用,因为这些字段必须在类实例化时实例化。声明不可变依赖项的惟一方法是使用基于构造器的依赖注入。

2.2 容易违反单一职责设计原则

在面向对象的编程中,五大设计原则SOLID被广泛应用,(国内一般为六大设计原则),用以提高代码的重用性,可读性,可靠性和可维护性

S在SOLID中代表单一职责原则,即即一个类应该只负责一项职责,这个类提供的所有服务都应该只为它负责的职责服务。

使用基于字段的依赖注入,高频使用的类随着时间的推移,我们会在类中逐渐添加越来越多的依赖项,我们用着很爽,很容易忽略类中的依赖已经太多了。但是如果使用基于构造函数的依赖注入,随着越来越多的依赖项被添加到类中,构造函数会变得越来越大,我们一眼就可以察觉到哪里不对劲。

有一个有超过10个参数的构造函数是一个明显的信号,表明类已经转变一个大而全的功能合集,需要将类分割成更小、更容易维护的块。

因此,尽管属性注入并不是破坏单一责任原则的直接原因,但它隐藏了信号,使我们很容易忽略这些信号。

2.3 与依赖注入容器紧密耦合

使用基于字段的依赖注入的主要原因是为了避免getter和setter的样板代码或为类创建构造函数。最后,这意味着设置这些字段的唯一方法是通过Spring容器实例化类并使用反射注入它们,否则字段将保持null。

依赖注入设计模式将类依赖项的创建与类本身分离开来,并将此责任转移到类注入容器,从而允许程序设计解耦,并遵循单一职责和依赖项倒置原则(同样可靠)。因此,通过自动装配(autowiring)字段来实现的类的解耦,最终会因为再次与类注入容器(在本例中是Spring)耦合而丢失,从而使类在Spring容器之外变得无用。

这意味着,如果您想在应用程序容器之外使用您的类,例如用于单元测试,您将被迫使用Spring容器来实例化您的类,因为没有其他可能的方法(除了反射)来设置自动装配字段。

2.4 隐藏依赖关系

在使用依赖注入时,受影响的类应该使用公共接口清楚地公开这些依赖项,方法是在构造函数中公开所需的依赖项,或者使用方法(setter)公开可选的依赖项。当使用基于字段的依赖注入时,实质上是将这些依赖对外隐藏了。

三、解决方案

我一般比较懒,在项目中由@Autowired直接改为@Resource注解了,方法简单粗暴,哈哈

3.1 使用@Resource

将@Autowired注解改为@Resource注解,即可解决这个问题

【注解使用】使用@Autowired注解警告Field injection is not recommended_注解_02

3.2 使用于构造函数的依赖注入

下图中使用的是lombok注解的方式添加构造函数。

【注解使用】使用@Autowired注解警告Field injection is not recommended_java_03

 下图中使用的是显示的 方式添加构造函数。

【注解使用】使用@Autowired注解警告Field injection is not recommended_spring_04

3.3 使用于Setter的依赖注入

基于Setter的DI是由容器在调用无参数构造函数或无参数静态工厂方法实例化bean后,在bean上调用Setter方法来完成的。

下面的示例显示了一个只能使用纯setter注入进行依赖注入的类。这个类是传统的Java。它是一个POJO,不依赖于特定于容器的接口、基类或注释。

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...
}

3.4 个人总结

我们已经看到,基于字段的注入应该尽可能地避免,因为它有许多缺点,无论它看起来多么优雅。推荐的方法是使用基于构造函数和基于setter的依赖注入。对于必需的依赖,建议使用基于构造函数的注入,设置它们为不可变的,并防止它们为null。对于可选的依赖项,建议使用基于sett的注入。对于这一点Spring官方文档也给出了相关的说明:Constructor-based or setter-based DI?

四、参考文档

1、Field injection is not recommended – Spring IOC:Field injection is not recommended – Spring IOC - Marc Nuri

2、Spring官方文档: Core Technologies

3、Field injection is not recommended(不再推荐使用字段注入):Field injection is not recommended(不再推荐使用字段注入) - 知乎

完结!


标签:依赖,Autowired,Spring,Field,字段,使用,注解,注入,构造函数
From: https://blog.51cto.com/u_16128050/6345342

相关文章

  • hibernate注解(一)
     在注解大行其道的今天,xml配置方式已经渐渐退化为非主流了,虽然注解有着很多优点,如:简化配置、减少配置文件数量、提高代码可读性,但它仍然有着无法媲美xml的地方。注解虽然配置简单易懂,但是对于复杂点的数据结构,配置起来反而比xml难(例如用xml配置,因为有帮助文件dtd的支持,因此能很......
  • hibernate注解(二)
    本篇将向大家介绍表关联的注解方式。1.一对一(One-to-one)使用@OneToOne注解可以建立实体bean之间的一对一的关联.一对一关联有三种情况:一是关联的实体都共享同样的主键,二是其中一个实体通过外键关联到另一个实体的主键(注意要模拟一对一关联必须在外键列上添加唯一约束).三是......
  • Springboot @Value注解
    配置文件test:name:123list:1,2,3aa:userInfoServiceImpl.merChantNoController:@Value("${test.name}")publicStringname1;//输出123@Value("#{'${test.list}'}")publicList<Object>list;//输出[1,2,3]......
  • 【SpringBoot】SpringBoot常用注解
    一、前言首先这里说的SpringBoot常用注解是指在我们开发项目过程中,我们经常使用的注解,包含Spring、SpringBoot、SpringCloud、SpringMVC等这些框架中的注解,而不仅仅是SpringBoot中的注解。这里只是作一个注解列举,每个注解具体如何使用可以自行搜索查询哈。二、配置启动相关注解2.1......
  • 注解,转发和重定向
    1注解1.1元注解:元注解是给注解加的注解@Target@Retention@Documented@Inherited@target......
  • Java注解
    注解的理解(1)、注解也被称为元数据,用于修饰解释包、类、方法、属性、构造器、局部变量等数据信息,附加在代码中的一些元信息(2)、和注释一样,注解不影响程序的逻辑,但注解可以被编译或运行,相当于嵌在代码中的补充信息@Overried(1)、Override只能用在方法上,用于重写父类的方法(可以是直......
  • springboot常用注解
    SpringBoot是基于Spring框架的部分组件进行封装,所以大部分注解都是Spring框架中常用的注解。下面是SpringBoot中常用的注解:-----------------------------------------------------------------------1、@SpringBootApplication:这个注解是SpringBoot应用的入口标志,......
  • java 反射:类和属性是否有注解
    booleanisAnnotationPresent(Class<?extendsAnnotation>annotationClass)元素上是否包含指定类型的注解,存在则返回true,否则返回false<AextendsAnnotation>AgetAnnotation(Class<A>annotationClass)获取元素上指定的注解,如果元素没有该注解返回null<AextendsAnn......
  • Spring MVC +MyBatis3全注解实例详解
    SpringMVC3.0.5+Spring3.0.5+MyBatis3.0.4全注解实例详解(一):[url]http://www.blogjava.net/bolo/archive/2011/05/23/349655.html[/url]如何配置Eclipse,Maven,Jetty并运行工程.如是使用Tomcat的话,改插件为:[url]http://tomcat.apache.org/maven-plugin......
  • 自定义注解实现数据序列化时进行数据脱敏(基于springboot默认jackjson)、消息转换器
    消息转换器fastjson与jackjson问题在springboot中使用fastjson的@jsonField无效原因:在springboot默认有json(jackjson)解析工具,所以使用fastjson不会生效解决方案替换默认的解析工具(笔者不推荐,这里根据自己项目决定)fastjson替换默认的jackjson第一种方法bean方法packagecom......