清洁单元测试

使用JUnit和一些模拟库编写“单元测试”测试很容易。 即使测试甚至不是单元测试并提供可疑的价值,它们也可能产生使某些涉众满意的代码覆盖范围。 编写单元测试(在理论上是单元测试,但比基础代码更复杂)因此也很容易编写,因此只会增加整个软件的熵。

这种特殊类型的软件熵具有令人不愉快的特征,这使得底层软件的重组或满足新需求变得更加困难。 就像测试具有负值一样。

正确地进行单元测试比人们想象的要困难得多。 在本文中,我概述了一些旨在提高单元测试的可读性,可维护性和质量的技巧。

注意:对于代码段,使用Spock。 对于那些不了解Spock的人,可以认为它是围绕JUnit的非常强大的DSL,它添加了一些不错的功能并减少了冗长性。

失败原因

仅当被测代码存在问题时,单元测试才应该失败。 仅当DBService存在错误时,才对DBService类的单元测试失败,而不是它依赖的任何其他类都存在错误时,则该单元测试应该失败。 因此,在DBService的单元测试中,唯一实例化的对象应该是DBService。 DBService依赖的所有其他对象都应该被存根或模拟。

否则,您将测试DBService以外的代码。 尽管您可能错误地认为这更划算,但这意味着定位问题的根本原因将需要更长的时间。 如果测试失败,则可能是因为多个类存在问题,但您不知道哪个类。 而如果仅由于被测代码错误而导致失败,则您可以确切地知道问题出在哪里。 此外,以这种方式思考将改善代码的面向对象性质。 测试仅测试班级的职责。 如果职责不明确,或者没有另一个类就不能做任何事情,或者该类太琐碎,测试是没有意义的,那么就会提示这个问题,即该类在其职责的一般性方面存在问题。 不模拟或存根依赖类的唯一例外是,如果您正在使用Java库中的知名类,例如String。 存根或嘲笑没有什么意义。 或者,从属类只是一个简单的不可变POJO,在其中没有存根或模拟它的价值。

存根和嘲笑

术语嘲笑和存根经常可以互换使用,就好像存在同一件事一样。 它们不是同一件事。 总而言之,如果您的被测试代码依赖于某个对象,而该对象从未在该对象上调用具有副作用的方法,则应将该对象存根。

而如果它依赖某个对象,并且确实为其调用了具有副作用的方法,则应该对其进行模拟。 为什么这很重要? 因为您的测试应根据与其依赖关系之间的关系类型来检查不同的事物。 假设您要测试的对象是BusinessDelegate。 BusinessDelegate接收编辑BusinessEntities的请求。 它执行一些简单的业务逻辑,然后在DBFacade(数据库前面的Facade类)上调用方法。 因此,正在测试的代码如下所示:

 public class BusinessDelegate { private DBFacade dbFacade; // ... public void edit(BusinessEntity businessEntity) { // Read some attributes on the business entity String newValue = businessEntity.getValue();       // Some Business Logic, Data Mapping, and / or Validation //... dbFacade.update(index, data) }  } 

关于BusinessDelegate类,我们可以看到两个关系。 与BusinessEntity的只读关系。 BusinessDelegate在其上调用一些getters(),并且从不更改其状态或调用任何具有副作用的方法。 与DBFacade的关系,它要求DBFacade做一些我们认为会做的事情。 确保更新发生不是BusinessDelegate的责任,这是DBFacade的工作。 BusinessDelegate的责任是确保仅使用正确的参数来调用更新方法。 很清楚,在对BusinessDelegate进行单元测试时,应将BusinessEntity存根,并应模拟DbFacade。 如果我们使用Spock测试框架,我们可以很清楚地看到这一点

 class BusinessDelegateSpec { @Subject BusinessDelegate businessDelegate def dbFacade def setup() { dbFacade = Mock(DbFacade) businessDelegate = new BusinessDelegate(dbFacade); } def "edit(BusinessEntity businessEntity)" () { given: def businessEntity = Stub(BusinessEntity) // ... when: businessDelegate.edit(businessEntity) then : 1 * dbFacade.update(data) }  } 

对存根模拟差异的深入了解可以极大地提高OO质量。 与其仅考虑对象的作用,不如考虑它们之间的关系和依赖性。 现在,单元测试可以帮助实施可能会迷失的设计原理。

存根和模拟在正确的位置

你们中的好奇者可能想知道为什么在上面的代码sampledbFacade中在类级别声明了而businessEntity在方法级声明了吗? 好吧,答案是,单元测试代码可读性越强,它越能反映被测代码。 在实际的BusinessDelegate类中,对dbFacade的依赖关系在类级别,而对BusinessEntity的依赖关系在方法级别。

在现实世界中,当实例化BusinessDelegate时,将存在DbFacade依赖关系,每当实例化BusinessDelegate进行单元测试时,也可以存在DbFacade依赖关系。 听起来合理吗? 希望如此。 这样做还有两个优点:

  • 减少代码详细程度。 即使使用Spock,单元测试也可能变得冗长。 如果将类级别的依赖关系移出单元测试,则将减少测试代码的冗长性。 如果您的班级在班级级别上依赖于其他四个班级,则每个测试中至少要包含四行代码。
  • 一致性。 开发人员倾向于以自己的方式编写单元测试。 如果他们是唯一阅读其代码的人,那就很好; 但是这种情况很少。 因此,我们在所有测试中拥有的一致性越强,维护起来就越容易。 因此,如果您读过从未读过的测试,并且至少看到由于特定原因而在特定位置对变量进行了打桩和模拟,那么您会发现单元测试代码更易于阅读。

变量声明顺序

这是最后一点的后续内容。 在正确的位置声明变量是一个很好的开始,下一步是按照它们在代码中出现的顺序进行操作。 所以,如果我们有类似下面的内容。

 public class BusinessDelegate { private BusinessEntityValidator businessEntityValidator; private DbFacade dbFacade; private ExcepctionHandler exceptionHandler; @Inject BusinessDelegate(BusinessEntityValidator businessEntityValidator, DbFacade dbFacade, ExcepctionHandler exceptionHandler) { // ... // ... } BusinessEntity read(Request request, Key key) { public BusinessEntity read(Request request, Key key) { // ... }      } 

如果它们的存根和模拟的定义与类声明它们的顺序相同,则读取测试代码要容易得多。

 class BusinessDelegateSpec { @Subject BusinessDelegate businessDelegate // class level dependencies in the same order def businessEntityValidator def dbFacade def exceptionHandler def setup() { businessEntityValidator = Stub(BusinessEntityValidator) dbFacade = Mock(DbFacade) exceptionHandler = Mock(ExceptionHandler) businessDelegate = new BusinessDelegate(businessEntityValidator, dbFacade, exceptionHandler) } def "read(Request request, Key key)" () { given: def request = Stub(Request) def key = Stub(key) when: businessDelegate. read (request, key) then : // ... }  } 

变量命名

而且,如果您认为最后一点是学究的,那么您也会很高兴知道这一点。 用于表示存根和模拟的变量名称应与实际代码中使用的名称相同。 更好的是,如果您可以在测试代码中将变量命名为与类型相同的名称,并且不会失去任何业务意义,则可以这样做。 在最后一个代码示例中,参数变量被命名为requestInfo和key,并且它们对应的存根具有相同的名称。 这比做这样的事情容易得多:

 //..  public void read(Request info, Key someKey) { // ...  } 
 // corresponding test code  def "read(Request request, Key key)" () { given: def aRequest = Stub(Request) def myKey = Stub(key) // you ill get dizzy soon! // ... 

避免过度存根

过多的存根(或嘲笑)通常意味着出现了问题。 让我们考虑一下得墨meter耳定律。 想象一下一些伸缩方法调用…

 List queryBusinessEntities(Request request, Params params) { // check params are allowed Params paramsToUpdate =       queryService.getParamResolver().getParamMapper().getParamComparator().compareParams(params) // ... // ...  } 

仅仅存根queryService是不够的。 现在,resolveAllowableParams()返回的任何内容都必须进行存根,并且该存根必须具有mapToBusinessParamsstubbed(),然后必须具有mapToComparableParams()。 即使使用Spock这样的框架,它可以最大限度地减少冗长,但对于一行Java代码,您将不得不进行四行存根。

 def "queryBusinessEntities()" () { given: def params = Stub(Params) def paramResolver = Stub(ParamResolver) queryService.getParamResolver() = paramResolver def paramMapper = Stub(ParamMapper) paramResolver.getParamMapper() >> paramMapper def paramComparator = Stub (ParamComparator) paramMapper.getParamComparator() >> paramComparator Params paramsToUpdate = Stub(Params) paramComparator.comparaParams(params) >> paramsToUpdate when: // ... then : // ...  } 

! 看看那一行Java对我们的单元测试的效果。 如果您不使用Spock之类的东西,情况会变得更糟。 解决方案是避免伸缩方法调用,并尝试仅使用直接依赖项。 在这种情况下,只需将ParamComparator直接注入到我们的类中即可。 然后代码变成…

 List queryBusinessEntities(Request request, Params params) { // check params are allowed Params paramsToUpdate = paramComparator.compareParams(params) // ... // ...  } 

测试代码变成

 setup() { // ... // ... paramComparator = Stub (ParamComparator) businessEntityDelegate = BusinessEntityDelegate(paramComparator)  }  def "queryBusinessEntities()" () { given: def params = Stub(Params) Params paramsToUpdate = Stub(Params) paramComparator.comparaParams(params) >> paramsToUpdate when: // .. then : // ...  } 

所有突然的人都应该感谢您减少头晕。

小黄瓜语法

不良的单元测试具有可怕的内容,例如断言在整个位置(顶部,中间和底部)。 它很快就会令人恶心,哪些是重要的,哪些是多余的。 哪些需要设置的位等等,等等。原理图更容易理解。 那是Gherkin语法的真正优势。 该场景是在给定的条件下设置的:总是,该场景何时出现,然后就是我们所期望的。 更好的用法是,像Spock这样的东西意味着您拥有一个漂亮,整洁的DSL,以便在给定的时间,然后在一个测试方法中将它们放在一起。

窄时宽然后

如果单元测试正在测试四种方法,那是单元测试吗? 考虑以下测试:

 def "test several methods" { given: // ... when: def name = personService.getname(); def dateOfBirth = personService.getDateOfBirth(); def country = personService.getCountry(); then : name == "tony" dateOfBirth == "1970-04-04" country == "Ireland"  } 

首先,如果Jenkins告诉您这失败了,那么您将必须扎根,找出班级的哪一部分是错误的。 因为测试不针对特定方法,所以您不会立即知道哪个方法失败。 其次,假设是getName()失败了,那么getDateOfBirth()和getCountry()如何工作? 测试在第一次失败时停止。 因此,当测试失败时,您甚至都不知道是有一种方法无效还是三种方法无效。 您可以四处告诉所有人您具有99%的代码覆盖率和一项测试失败。 但是-一项测试完成了多少?

此外,什么更容易解决? 小测试还是长测试? 理想情况下,测试应检查与您正在测试的事物之间的单个交互。 现在,这并不意味着您只能拥有一项资产,而是应该拥有一个狭窄的资产,然后再拥有一个广阔的资产。 因此,让我们先缩小一下范围。 理想情况下,仅一行代码。 一行代码与您要进行单元测试的方法匹配。

 def "getName()" { given: // ... when: def name = personService.getname(); then : name == "tony"  }  def "getDateOfBirth()" { given: // ... when: def dateOfBirth = personService.getDateOfBirth(); then : dateOfBirth == "1970-04-04"  }  def "getCountry()" { given: // ... when: def country = personService.getCountry(); then : country == "Ireland"  } 

现在,如果getName()失败,但getCountry()和getDateOfBirth()通过,则我们可以拥有完全相同的代码覆盖率,但是getName()而不是getCountry()和getDateOfBirth()出现了问题。 获得测试的粒度与代码覆盖率完全不同。 理想情况下,对于每种非私有方法,它应该至少是一个单元测试。 当您将否定测试等因素考虑在内时,效果会更好。在单元测试中具有多个断言是完全可以的。 例如,假设我们有一个委托给其他类的方法。

考虑一个resynceCache()方法,该方法在其实现中会在cacheService对象上调用另外两个方法:clear()和reload()。

 def "resyncCache()" { given: // ... when: personService.resyncCache(); then : 1 * cacheService. clear () 1 * cacheService.reload()  } 

在这种情况下,进行两个单独的测试是没有意义的。 “时间”相同,并且如果任何一个失败,您将立即知道必须查看哪种方法。 进行两次单独的测试仅意味着付出两倍的努力,却几乎没有收益。 要做的一个微妙的事情是确保您的资产顺序正确。 它们应与代码执行的顺序相同。 因此,在reload()之前调用clear()。 如果在clear()处测试失败,则由于方法被破坏,无论如何都没有必要检查reload()。 如果您不遵循断言顺序提示,而是先对reload()进行断言并且被报告为失败,那么您将不知道应该首先发生的clear()是否已经发生。 以这种方式思考将帮助您成为一名测试忍者!

嘲笑和存根的排序技巧也适用于断言。 按时间顺序断言。 这很花哨,但是它将使测试代码更易于维护。

参数化

参数化是一项非常强大的功能,可以大大降低测试代码的详细程度,并Swift增加代码路径中的分支覆盖范围。 单元测试忍者应该总是能够发现何时使用它!

一个明显的迹象表明,可以将多个测试分组为一个测试并对其进行参数化,即它们具有相同的when块,不同的输入参数除外。 例如,请考虑以下内容。

 def "addNumbers(), even numbers" () { given: // ... when: def answer = mathService.addNumbers(4, 4); then : // ...  }  def "addNumbers(), odd numbers" () { given: // ... when: def answer = mathService.addNumbers(5, 5); then : // ...  } 

正如我们在这里看到的,除了输入参数外,when相同。 这对于参数化毫无疑问。

 @Unroll( "number1=#number1, number2=#number2" ) // unroll will provide the exact values in test report unroll will provide the exact values  def "addNumbers()" (int number1, int number2) { given: // ... when: def answer = mathService.addNumbers(number1, number2); then : // ... where: number1  | number2  || answer 4        | 4        || 8 5        | 5        || 10  } 

立即我们将代码减少了50%。 通过将另外一行添加到where表中,我们还使添加其他排列变得更加容易。 因此,虽然看起来这两个测试应该是一个参数化测试非常明显,但是只有遵守时机狭窄的准则才是显而易见的。 狭窄的“何时”编码风格使要测试的确切场景更容易看到。 如果将广泛的时间用于很多事情,那么事实并非如此,因此很难发现要参数化的测试。

通常,唯一不对具有相同语法的测试进行参数化的时间是:代码块是指期望是完全不同的结构。 期望一个int是相同的结构,在一个场景中期望一个异常而一个int是另一个场景则是两个不同的结构。 在这种情况下,最好不要参数化。 一个经典的和众所周知的例子是正负测试的混合。 假设我们的addNumbers()方法在接收到浮动后会抛出异常,这是一个否定的测试,应该分开放置。 then:块绝不能包含if语句。 这是测试变得越来越灵活的标志,而没有if语句的单独测试更有意义。

摘要

干净的单元测试对于拥有可维护的代码基础,能够定期且快速地发布并更享受您的软件工程至关重要。

翻译自: https://www.javacodegeeks.com/2020/03/clean-unit-testing.html

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

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

相关文章

谷歌发布android+8.2,谷歌Pixel 2代三款新机齐曝光:骁龙835+Android 8.0

随着5月份谷歌I/O大会的临近,关于Android O(安卓8.0)和Pixel手机的消息逐渐多了起来。据外媒报道, Android Open Source Project的代码表中确认,三款Pixel新机搭载的都是MSM8998芯片,也就是骁龙835。目前,Pixel 2的代号…

怎么停止skywalking_Skywalking部署常见问题以及注意事项

Skywalking部署常见问题以及注意事项IntroSkyWalking 创建与2015年,提供分布式追踪功能。从5.x开始,项目进化为一个完成功能的Application Performance Management系统。他被用于追踪、监控和诊断分布式系统,特别是使用微服务架构&#xff0c…

akka的介绍_Akka笔记–演员介绍

akka的介绍过去做过多线程的任何人都不会否认管理多线程应用程序有多么艰辛和痛苦。 我说管理是因为它开始很简单,一旦您开始看到性能改进,它就会变得非常有趣。 但是,当您发现没有一种简单的方法可以从子任务中的错误或难以发现的僵尸错误中…

HTML语言name,HTML a 标签的 name 属性

延伸阅读:使用 name 属性还是 id 属性?在 HTML 4.0 之前的版本中,只有使用 标签的 name 属性才能创建片段标识符。随着 HTML 4.0 中 id 属性的出现,所有 HTML 或 XHTML 元素都可以是片段标识符。这是因为 id 标识符几乎可以用在所…

使用模拟进行测试

如果使用正确的方法,模拟对象将非常有用。 我在需要驱动软件开发使用的帖子中分享了一些使用Mock Objects的经验。 在这篇文章中,我分享了两件事 –使用模拟进行基于合同的测试。 –用于组织模拟代码的模式。 基于合同的测试 让我们以正在构建汇款服…

hazelcast_Hazelcast的MapLoader陷阱

hazelcastHazelcast提供的核心数据结构之一是IMap<K, V> &#xff0c;它扩展了java.util.concurrent.ConcurrentMap &#xff0c;它基本上是一个分布式地图&#xff0c;通常用作缓存。 您可以将此类地图配置为使用自定义MapLoader<K, V> -每次尝试从该地图&#xf…

html ul左侧浮动,UL里的LI元素左浮动层一行显示时使其居中的方法

在制作页面是&#xff0c;li浮动是很常用的&#xff0c;一般情况也不用其居中&#xff0c;但有时&#xff0c;其特殊原因需要居中&#xff0c;这是就有点犯难了&#xff0c;这里有了一个很好的解决方法&#xff0c;主要是用了相对定位的原理。在ul外报一层&#xff0c;使其相对…

aws s3 獲取所有文件_AWS SA associate 证书考试学习记录-EBS,S3,EFS比较

我们的目标&#xff0c;就是花最少的时间&#xff0c;学到最多的东西&#xff1a;-&#xff09;在AWS中&#xff0c;可以选择的存储服务很多&#xff0c;纷繁复杂&#xff0c;新手根本弄不清楚选择哪个。因为做为一个云架构师&#xff0c;你并不是让你的产品能用就够了&#xf…

ios调用restful接口_做iOS上最好的REST API测试App

对于Web开发者和移动应用开发者来说&#xff0c;少不了和REST API打交道。何为REST API&#xff0c;维基百科是这么解释的(https://zh.wikipedia.org/wiki/REST)REST(英文&#xff1a;Representational State Transfer)是Roy Thomas Fielding博士于2000年在他的博士论文中提出来…

Java 14:记录

Java 14是在几周前问世的&#xff0c;它引入了Record类型&#xff0c;它是一个不变的数据载体类&#xff0c;旨在容纳一组固定的字段。 请注意&#xff0c;这是预览语言功能 &#xff0c;这意味着必须使用--enable-preview标志在Java编译器和运行时中显式启用它。 我将直接介绍…

html设置顶部对齐,HTML / CSS文本从div顶部对齐

首先&#xff0c;你需要修复你的CSS选择器 .你可以通过这种方式t write all those id .#content #main #services只需选择1个元素和他的孩子 .例如&#xff0c;如果你测试它&#xff0c;它将适用于你&#xff1a;#services .langelis .txt {width: 440px;height: auto;float: l…

java与java ee_计划Java EE 7批处理作业

java与java eeJava EE 7添加了使用JSR 352以标准方式执行批处理作业的功能。 <job id"myJob" xmlns"http://xmlns.jcp.org/xml/ns/javaee" version"1.0"><step id"myStep"><chunk item-count"3"><rea…

html5 video 播放状态,10分钟了解HTML5的Video标签属性、方法和事件

标签的属性src &#xff1a;视频的属性poster&#xff1a;视频封面&#xff0c;没有播放时显示的图片preload&#xff1a;预加载autoplay&#xff1a;自动播放loop&#xff1a;循环播放controls&#xff1a;浏览器自带的控制条width&#xff1a;视频宽度height&#xff1a;视频…

hot编码 字符one_One Hot编码是什么?为什么要用它,什么时候用它?

作者&#xff1a;Rakshith Vasudev编译&#xff1a;ronghuaiyang导读当你在玩ML模型的时候&#xff0c;你会在任何地方遇到这个“One hot encoding”的术语。当你在玩ML模型的时候&#xff0c;你会在任何地方遇到这个“One hot encoding”术语。你可以看到一个one hot编码器的s…

CUBA 7.2 –有什么新功能?

CUBA平台的第七版向前迈出了一大步。 内部体系结构的改进和新的IDE为进一步改进奠定了良好的基础。 我们将继续添加新功能&#xff0c;以使开发人员的生活更轻松&#xff0c;并使他们的工作更加高效。 在7.2版中&#xff0c;我们引入了许多可能看起来像是主要更新的更改&#…

postgresql 分区视图_PostgreSQL架构集中式到分布式主流架构总结

文章目录一、PG未来主流架构为什么是分布式二、PostgreSQL集中式到分布式架构总结一、PG未来主流架构为什么是分布式如果说5年前DB的分布式还只是一种趋势&#xff0c;如今分布式数据库正逐渐从趋势变成主流。说到分布式&#xff0c;我想我们不能不提一下集中式和分库分表。01集…

html5 上传图片模板,HTML5实现图片文件异步上传

&#xff0c;过现前个能文使近记接的端问对字用近记接  利用HTML5的新特点做文件异步上传非常简单方便&#xff0c;本文主要展示JS部分&#xff0c;html结构。下面的代码并未使用第三发库&#xff0c;如果有参照&#xff0c;请注意一些未展现出来的代码片段。我这边的效果预览…

dot2谜团png_一个类加载的谜团解决了

dot2谜团png面对一个好老问题 我在应用程序服务器上遇到一些类加载问题。 这些库被定义为Maven依赖项&#xff0c;因此被打包到WAR和EAR文件中。 不幸的是&#xff0c;其中一些还安装在应用程序服务器中&#xff0c;但版本不同。 启动应用程序时&#xff0c;我们遇到了与这些类…

python log文件如何不写入syslog_Centos下python 对syslog重写进行日志记录

在Linux 环境下&#xff0c;python自带一个syslog的模块可以进行日志记录。python可以利用logging模块来重写syslog&#xff0c;这样就可以自定义写入文件的文件名。如果不做配置则直接写入到/var/log/message文件里。首先先写log.py,代码如下&#xff1a;importosimportsysimp…

html中可以有两个h1,在一个HTML中h1标签能出现几次?h1标签和标题标签

首页 > web前端 > html教程 > 正文 在一个HTML中h1标签能出现几次&#xff1f;h1标签和标题标签的差别是什么&#xff1f; 2018-08-29 10:57:28本篇文章主要介绍了关于HTML h1标签的一些解释&#xff0c;有html h1标签和html title标签的区别&#xff0c;还有网页中h1…