依赖注入–字段vs构造函数vs方法

嗨,今天,我想简单地讨论将依赖项注入您的类的不同方式。

通常,您有以下三种注射方法

  • 直接进入字段/属性
  • 通过显式的setter方法
  • 通过显式的构造函数参数


现场注入

这种类型的注入为将所需的依赖项注入到类中提供了某种反射机制。

尽管这种注入类型的好处是,它消除了诸如setter方法或构造函数参数之类的混乱代码,但其缺点是这些依赖性是不可见的。 如果从外部查看类,则只会看到公共方法,并且可能是构造函数。

即使这使您非常清楚地了解了类提供的服务,但在我看来,它仍然具有以下主要缺点:

在为此特定类编写测试时,您必须检查该类以查看所需的依赖项,并且必须使用DI框架(即使是简单的测试),也必须使用一种反射机制来注入依赖项(模拟/存根/真实)。

更糟糕的是,传入依赖项的数量隐藏在该类中。 当然,您可以使用工具(JDepend等,pp。)告诉您依赖性的数量和方向,或者使用一个文件来指定注入,但是您必须依靠这种工具或检查类。

我观察到的另一个缺点是,与使用Setter或Constructor Injection的情况相比,创建具有多个职责的类的机会更高。
就像是:

哦,让我们使用这个精美的注解在这里注入我们需要的服务...几天/几小时后:测试这种野兽是如此的困难

二传手注射

注入器注入工具注入器方法,每个依赖项一个,DI框架用于注入依赖项。

这是一个变体,它使依赖项显式显示,并且使您可以清楚地了解特定类的依赖项。

在测试过程中,它的好处是您不必使用DI框架或反射机制,而可以直接设置依赖项。

这种方法的缺点是:您可以构造处于无法使用状态的类。 这是因为如果需要或可选依赖项,则无法与外部区分开。

构造函数注入

构造函数注入对类的构造函数进行检测,DI框架使用该类来注入依赖项。 这是使依赖关系明确的另一个变体。

与Setter注入相反,它可以防止您在不规则状态下创建类。 (当然,您可以传递null,但这不仅仅是一种作弊,对吗?)所以我要说,这是最严格的变体:

每个依赖项都是强制性的

这种注射类型的好处是

  1. 您必须只阅读一种方法,即构造函数,才能弄清楚此类的依赖关系。
  2. 您创建了一个不可变的类,它使缓存变得如此简单

同样,这里的缺点是您无法区分可选依赖项和必需依赖项。 构造函数强制设置所有字段。

我要讨论的最后一个变体是:

混合二传手和构造剂注射

我个人更喜欢此变体,混合了Setter和Constructor注入。

这给您:

  • 在合同级别上区分强制性依赖项和可选性的能力
  • 清楚说明特定类的依赖项是什么
  • 使您可以轻松地检查班级是否负有重大责任
  • 构建后正确配置的类

与纯方法类似,使用这种注入进行测试非常简单。 您不需要DI框架,并且可以轻松地将模拟的/存根的/真实的实现传递到要测试的类中。

使用/喜欢什么?

这个答案在很大程度上取决于您的框架/团队规则/语言。

但是我强烈建议您使用显式注入变量之一,因为它们使您无需使用依赖注入框架即可编写纯单元测试。

翻译自: https://www.javacodegeeks.com/2015/01/dependency-injection-field-vs-constructor-vs-method.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/361011.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Java 8 Streams API作为友好的ForkJoinPool外观

我最喜欢Java 8的功能之一是流API。 最终,它消除了代码中的几乎所有循环,并使您可以编写更具表现力和重点的代码。 今天,我意识到它可以用于其他用途:作为ForkJoinPool一个不错的前端。 问题:执行器样板 假设我们要并…

(转)ZwQuerySystemInformation枚举内核模块及简单应用

http://hi.baidu.com/_achillis/item/8b33ead8ccac28ea3cc2cb17 简单说,即调用第11号功能,枚举一下内核中已加载的模块。部分代码如下://功能号为11,先获取所需的缓冲区大小ZwQuerySystemInformation(SystemModuleInformation,NUL…

在三个Java IDE中生成的三种常见方法

在本文中,我研究了NetBeans 8.0.2 , IntelliJ IDEA 14.0.2和Eclipse Luna 4.4.1生成的三种“通用”方法[ equals(Object) , hashCode()和toString() ]的区别 。 。 目的不…

Angularjs总结(五)指令运用及常用控件的赋值操作

1、常用指令 1 <div ng-controller"jsyd-controller">2 <div style"float:left;width:100%; " ng-show"clickValue1登记">3 4 <div ng-include src"/partials/11.html"></div>5 6 </div&g…

linux截图软件

在Linux下很多软件使用命令就可以很好的操作&#xff0c;截图软件也不例外。刚好要截图使用一下&#xff0c;就找到了这款小巧的却很使用的Linux下的截图工具&#xff0c;就是scrot。 如何安装及使用? 1.安装篇很简单&#xff0c;想安装其他软件一样&#xff0c;Fedora下 yum …

Web开发的那点事--软件复用

CSDN博客不再经常更新&#xff0c;更多优质文章请来 粉丝联盟网 FansUnion.cn! (FansUnion) 复用的战场 1.前台 CSS,JavaScript/jquery/AJAX HTML/JSP 2.后台 增删改查 几乎一样。复用的级别 代码&#xff1a;一行代码或几行代码 函数&#xff1a;一个函数 类&#xff1a…

NodeJS学习笔记—1.CommonJS规范

由于现在web开发&#xff0c;越来越重视代码的复用和抽象的封装&#xff0c;为了解决代码的组织结构、管理、复用和部署等问题&#xff0c;现在普遍采用的机制是模块机制&#xff08;module&#xff09;。CommonJS约定桌面应用程序和服务器应用程序需要的API&#xff0c;如操作…

鼠标事件和键盘事件总结 及判断是不是数字方法

事件 Delegate 命名空间 数据的类 实现 鼠 标 事 件 "MouseHover" "MouseLeave" "MouseEnter" EventHandler System EventArgs 1、定义&#xff1a;"组件名"."事件名称" new System.EventHandl…

angularjs 利用filter进行表单查询及分页查询

页面&#xff1a; <div><input style"width:90%;margin-left:5px;margin-right:5px;" class"form-control sys_input" ng-model"imagePaths.filter.imageName" placeholder"查询..."/></div><div><!--<…

为什么现在是升级到Java 8的最佳时机

有兴趣了解如何通过AppDynamics充分利用Java 8的新功能吗&#xff1f; 立即开始免费试用 &#xff01; 今年3月&#xff0c;Oracle发布了近十年来最受期待的版本Java8。自发布以来&#xff0c;最新版本引起了越来越多的关注&#xff0c;各种规模的公司都渴望升级。 我们的合作…

requirejs与anjularjs框架

1.目录 2.首页login.html如下&#xff1a; <!DOCTYPE html><html> <head> <title>登录界面</title> <link relstylesheet href/stylesheets/style.css /> <link rel"stylesheet" href"/css/bootstrap.min.css">…

Qt基于TCP网络程序发包封包抽象

之前没经验, 发送数据包的时候, 包头包尾等信息都是通过重新定义一个结构体实现, 不同的协议包就有不同的结构体, 结果导致这样的现象: 有多少上层业务协议包, 我就分别重新定义一个对应的加上包头包尾的新的结构体, 很费劲.......额.... 现在, 重新想了下, 一个改进的方式, 把…

SQL学习笔记

可以把 SQL 分为两个部分&#xff1a;数据操作语言 (DML) 和 数据定义语言 (DDL)。 查询和更新指令构成了 SQL 的 DML 部分&#xff1a; SELECT - 从数据库表中获取数据UPDATE - 更新数据库表中的数据DELETE - 从数据库表中删除数据INSERT INTO - 向数据库表中插入数据SQL 的数…

Spring MVC 4快速入门Maven原型已改进

Spring Boot使Spring入门非常容易。 但是仍然有人对不使用Spring Boot并以更经典的方式引导应用程序感兴趣。 几年前&#xff0c;我创建了一个原型&#xff08;早于Spring Boot&#xff09;&#xff0c;简化了引导Spring Web应用程序的过程。 尽管Spring Boot已经上市了一段时间…

图片循环播放

使用 pageSwitch插件 多种效果 引入 jquery.js 和 pageSwitch.min.js <script src"js/jquery-1.11.0.min.js"></script> <script src"dist/pageSwitch.min.js"></script>在页面定义标签 <div id"container">…

当你辛辛苦苦写的博客文章被无情复制,成为了他的原创,你作何感想?

我一直都说我之所以开始写博客&#xff0c;是因为我是想把博客当成一个备忘录&#xff0c;同时也能分享给大家。我才开博1个月&#xff0c;还没有写几篇文章&#xff0c;我发现每篇文章都被很多网站转载了&#xff0c;有的署名或者是贴上我文章的地址作为来源地址。 对于这种情…

SSL / TLS REST服务器–带有Spring和TomEE的客户端

在构建系统时&#xff0c;开发人员通常会忽略安全性方面。 安全一直是令人担忧的重要问题&#xff0c;但是它比以前吸引了更高的关注。 就在今年&#xff0c;我们发生了像Heartbleed Bug或CelebrityGate丑闻这样的案件。 这与帖子无关&#xff0c;只是安全真正重要的示例&#…

linux apf防火墙安装配置

linux apf防火墙安装配置APF(Advanced Policy Firewall)是 Rf-x Networks 出品的Linux环境下的软件防火墙,被大部分Linux服务器管理员所采用,使用iptables的规则,易于理解及使用。 www.2cto.com 适合对iptables不是很熟悉的人使用&#xff0c;因为它的安装配置比较简单&#x…

jquery实现上传图片及图片大小验证、图片预览效果代码

jquery实现上传图片及图片大小验证、图片预览效果代码jquery实现上传图片及图片大小验证、图片预览效果代码 上传图片验证 */ function submit_upload_picture(){ var file $(file_c).value; if(!/.(gif|jpg|jpeg|png|gif|jpg|png)$/.test(file)){ aler…

使用Spring Boot和Logback登录到Redis

在进行集中式日志记录时&#xff0c;例如使用Elasticsearch&#xff0c;Logstash和Kibana或Graylog2&#xff0c;您可以为Java应用程序提供几个选项。 您既可以编写标准的应用程序日志&#xff0c;也可以使用Logstash解析这些日志&#xff0c;这些日志既可以直接使用&#xff0…