热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

别再用Field注入了

Spring依赖注入先看看Spring的几种依赖注入方式:构造器注入、Field注入和Setter注入。构造器注入将被依赖对象通过构造函数的参数注入给依赖对象&

Spring依赖注入

先看看Spring的几种依赖注入方式:构造器注入、Field 注入和Setter注入。

构造器注入

将被依赖对象通过构造函数的参数注入给依赖对象,并且在初始化对象的时候注入。Spring 推荐的注入方式,适合强制依赖用法。fi

private DependencyA dependencyA;
private DependencyB dependencyB;;// @Autowired不是必须的,在 Spring4.x 中增加了新的特性:如果类只提供了一个带参数的构造方法,则不需要对对其内部的属性写 @Autowired 注解,Spring 会自动为你注入属性。
@Autowired
public DI(DependencyA dependencyA, DependencyB dependencyB) {this.dependencyA = dependencyA;this.dependencyB = dependencyB;
}

  • 优点: 对象初始化完成后便可获得可使用的对象,单元测试使用 Mock 就可以无需启动 DI 容器就可以实例化。
  • 缺点: 当需要注入的对象很多时,构造器参数列表将会很长;不够灵活。若有多种注入方式,每种方式只需注入指定几个依赖,那么就需要提供多个重载的构造函数。

Setter注入

IoC Service Provider 通过调用成员变量提供的 setter 函数将被依赖对象注入给依赖类。如果有可选可变的依赖就使用 setter 注入,而且可用 @Autowired(required = false) 来指定可选依赖项,构造注入则不能这么干,因为是应用于所有构造函数。

private DependencyA dependencyA;
private DependencyB dependencyB;@Autowired
public void setDependencyA(DependencyA dependencyA) {this.dependencyA = dependencyA;
}@Autowired(required = false)
public void setDependencyB(DependencyB dependencyB) {this.dependencyB = dependencyB;
}

setter 注入比较灵活,可以选择性地注入需要的对象。

Field注入

属性注入,在 bean 变量上使用注解进行依赖注入,本质上是通过反射的方式直接注入到 field。这应该是平时开发见到最多的的一种方式。

@Autowired
private DependencyA dependencyA;@Autowired
private DependencyB dependencyB;

Filed 注入是最简单方便的方式。

Field注入的局限和替代方案


局限

在以往Field注入应该是我们用的最多的依赖注入方式,直接引入bean然后在变量上使用@Autowired就行了,不过后来你会发现在IDEA会给出警告“Field injection is not recommend”。

由于Field注入使用简单,我们有意无意就会引入许多依赖,当注入依赖太多的时候意味着类承担了太多的责任,这违反面向对象的单一职责原则,而且不管引入多少都没有警告,因为这种方式可以无限扩展。

不过Field注入也有局限,无法在声明为final/immutable的字段上使用@Autowired注解,因为这些字段必须在类实例化的时候被初始化。

再者,Field注入对单元测试也不友好,你不得不使用Spring IoC容器来创建这些bean(和IoC容器强耦合了),但是单元测试原则上要快,启动IoC容器太慢,如果是构造注入的话,我们完成可以bean当成一个普通类来创建对象,直接通过构造传入就行。

替代方案

如果不使用@Autowired,我们还可以使用JDK提供的@Resource注解,可减少与Spring的耦合,使用一样简单,所以一样会有容易滥用的问题。

更推荐的方式是使用构造注入。当需要越多依赖的时候,构造参数越多,看起来很丑陋,我们可以使用Lombok来简化构造器注入。

Lombok提供了三个相关的注解来简化依赖注入:

  • @AllArgsConstructor 用来生成包含所有字段构的造方法;

  • @NoArgsConstructor 用来生成无参的构造方法;

  • @RequiredArgsConstructor 生成的构造方法只包含声明为final或者non-null的字段。

然后你会发现使用变得和Field注入差不多简单,想防止滥用还是得自己控制,把类设计好,避免包含太多职责。


原文链接。访问我的个人网站查看更多文章。


推荐阅读
  • Spring常用注解(绝对经典),全靠这份Java知识点PDF大全
    本文介绍了Spring常用注解和注入bean的注解,包括@Bean、@Autowired、@Inject等,同时提供了一个Java知识点PDF大全的资源链接。其中详细介绍了ColorFactoryBean的使用,以及@Autowired和@Inject的区别和用法。此外,还提到了@Required属性的配置和使用。 ... [详细]
  • 本文介绍了如何使用Express App提供静态文件,同时提到了一些不需要使用的文件,如package.json和/.ssh/known_hosts,并解释了为什么app.get('*')无法捕获所有请求以及为什么app.use(express.static(__dirname))可能会提供不需要的文件。 ... [详细]
  • Java学习笔记之面向对象编程(OOP)
    本文介绍了Java学习笔记中的面向对象编程(OOP)内容,包括OOP的三大特性(封装、继承、多态)和五大原则(单一职责原则、开放封闭原则、里式替换原则、依赖倒置原则)。通过学习OOP,可以提高代码复用性、拓展性和安全性。 ... [详细]
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • 基于PgpoolII的PostgreSQL集群安装与配置教程
    本文介绍了基于PgpoolII的PostgreSQL集群的安装与配置教程。Pgpool-II是一个位于PostgreSQL服务器和PostgreSQL数据库客户端之间的中间件,提供了连接池、复制、负载均衡、缓存、看门狗、限制链接等功能,可以用于搭建高可用的PostgreSQL集群。文章详细介绍了通过yum安装Pgpool-II的步骤,并提供了相关的官方参考地址。 ... [详细]
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • 如何使用Java获取服务器硬件信息和磁盘负载率
    本文介绍了使用Java编程语言获取服务器硬件信息和磁盘负载率的方法。首先在远程服务器上搭建一个支持服务端语言的HTTP服务,并获取服务器的磁盘信息,并将结果输出。然后在本地使用JS编写一个AJAX脚本,远程请求服务端的程序,得到结果并展示给用户。其中还介绍了如何提取硬盘序列号的方法。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • 本文介绍了九度OnlineJudge中的1002题目“Grading”的解决方法。该题目要求设计一个公平的评分过程,将每个考题分配给3个独立的专家,如果他们的评分不一致,则需要请一位裁判做出最终决定。文章详细描述了评分规则,并给出了解决该问题的程序。 ... [详细]
  • 阿,里,云,物,联网,net,core,客户端,czgl,aliiotclient, ... [详细]
  • Spring特性实现接口多类的动态调用详解
    本文详细介绍了如何使用Spring特性实现接口多类的动态调用。通过对Spring IoC容器的基础类BeanFactory和ApplicationContext的介绍,以及getBeansOfType方法的应用,解决了在实际工作中遇到的接口及多个实现类的问题。同时,文章还提到了SPI使用的不便之处,并介绍了借助ApplicationContext实现需求的方法。阅读本文,你将了解到Spring特性的实现原理和实际应用方式。 ... [详细]
  • Python SQLAlchemy库的使用方法详解
    本文详细介绍了Python中使用SQLAlchemy库的方法。首先对SQLAlchemy进行了简介,包括其定义、适用的数据库类型等。然后讨论了SQLAlchemy提供的两种主要使用模式,即SQL表达式语言和ORM。针对不同的需求,给出了选择哪种模式的建议。最后,介绍了连接数据库的方法,包括创建SQLAlchemy引擎和执行SQL语句的接口。 ... [详细]
  • 预备知识可参考我整理的博客Windows编程之线程:https:www.cnblogs.comZhuSenlinp16662075.htmlWindows编程之线程同步:https ... [详细]
  • position属性absolute与relative的区别和用法详解
    本文详细解读了CSS中的position属性absolute和relative的区别和用法。通过解释绝对定位和相对定位的含义,以及配合TOP、RIGHT、BOTTOM、LEFT进行定位的方式,说明了它们的特性和能够实现的效果。同时指出了在网页居中时使用Absolute可能会出错的原因,即以浏览器左上角为原始点进行定位,不会随着分辨率的变化而变化位置。最后总结了一些使用这两个属性的技巧。 ... [详细]
  • 如何查询zone下的表的信息
    本文介绍了如何通过TcaplusDB知识库查询zone下的表的信息。包括请求地址、GET请求参数说明、返回参数说明等内容。通过curl方法发起请求,并提供了请求示例。 ... [详细]
author-avatar
campionezhao
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有