JUnit 5 –扩展模型

我们已经对Java最普遍的测试框架的下一个版本了解很多。 现在让我们看一下JUnit 5扩展模型,该模型将允许库和框架将自己的实现添加到JUnit中。

总览

  • 设定
  • 基本
  • 建筑
  • 扩展模型
  • 条件
  • 注射

在新兴的《 JUnit 5用户指南》中可以找到您将在此处阅读的更多内容以及更多内容。 请注意,它基于Alpha版本,因此可能会发生变化。

确实,我们鼓励我们提出问题或提出请求,以便JUnit 5可以进一步改进。 请利用这个机会! 这是我们帮助JUnit帮助我们的机会,因此,如果您能在这里看到一些改善,请确保将其上游 。

如有必要,此帖子将得到更新。 我在这里显示的代码示例可以在GitHub上找到 。

JUnit 4扩展模型

首先让我们看一下JUnit 4是如何解决该问题的。 它具有两个部分竞争的扩展机制:运行程序和规则。

跑步者

测试运行者管理测试的生命周期:实例化,调用设置和拆卸方法,运行测试,处理异常,发送通知等。而JUnit 4提供了实现所有这些功能的实现。

在4.0中,只有一种扩展JUnit的方法:创建一个新的运行器并使用@RunWith(MyRunner.class)注释测试类,以便JUnit使用它而不是其自己的实现。

该机制非常繁重,并且不方便扩展。 而且它有一个非常严格的限制:每个测试班级只能有一位跑步者,这使得他们无法组成。 因此,无法同时利用Mockito和Spring跑步者的功能。

规则

为了克服这些限制,JUnit 4.7引入了rules ,它们是测试类的带注释字段。 JUnit 4将测试方法(和其他操作)包装到一条语句中,并将其传递给规则。 然后,他们可以在执行语句之前和之后执行一些代码。 此外,测试方法通常在执行期间在规则实例上调用方法。

一个示例是临时文件夹规则 :

public static class HasTempFolder {@Rulepublic TemporaryFolder folder= new TemporaryFolder();@Testpublic void testUsingTempFolder() throws IOException {File createdFile= folder.newFile("myfile.txt");File createdFolder= folder.newFolder("subfolder");// ...}
}

由于使用@Rule批注,JUnit调用带有包装方法testUsingTempFolder的语句的文件夹 。 编写此特定规则是为了使文件夹创建一个临时文件夹,执行测试,然后再删除该文件夹。 然后,测试本身可以在临时文件夹中创建文件和文件夹。

其他规则可能会在Swing的事件分发线程中运行测试 ,建立和删除数据库,或者如果测试运行时间过长,则让测试超时 。

规则是一个很大的改进,但通常仅限于在测试运行之前和之后执行一些代码。 他们无法帮助无法在该框架内实现的扩展。

事态

JUnit有两种相互竞争的扩展机制,每种都有其自身的局限性。

因此,自JUnit 4.7起就有两种竞争的扩展机制,每种机制都有其自身的局限性,但也有很多重叠之处。 这使得干净扩展很困难。 此外,编写不同的扩展可能会出现问题,并且通常无法实现开发人员希望的那样。

junit-5-extension-model

由Tony Walmsley在CC-BY 2.0下发布

JUnit 5扩展模型

JUnit Lambda项目具有几个核心原则 ,其中之一是“优先于功能而不是扩展点”。 从字面上看,这转化为新版本的整体机制–这不仅是扩展JUnit 5的唯一机制,也是最重要的机制。

延伸点

JUnit 5扩展可以声明对测试生命周期的某些特定时刻感兴趣。 当JUnit 5引擎处理测试时,它将逐步通过这些步骤并调用每个已注册的扩展。 从外观上看,这些是扩展点:

  • 测试实例后处理
  • 之前回调
  • 有条件的测试执行
  • 每次回调之前
  • 参数解析
  • 异常处理
  • AfterEach回调
  • 毕竟回调

(不必担心它们是否每个都还不清楚。我们稍后将介绍其中的一些。)

每个扩展点对应一个接口。 他们的方法采用的参数可以捕获测试生命周期中特定点的上下文,例如测试实例和方法,测试名称,参数,注释等。

扩展可以实现任意数量的那些接口,并且将由引擎使用相应的参数进行调用。 然后,它可以执行实现其功能所需的任何操作。 要考虑的一个细节:引擎在实例化扩展时以及实例保持存在的时间方面不做任何保证,因此它们必须是无状态的。 他们需要维护的任何状态都必须写入JUnit并从中加载。

创建扩展后,剩下要做的就是告诉JUnit。 这是那么容易,因为添加@ExtendWith(MyExtension。 ),需要延长测试类或方法。

实际上,存在一个稍微不那么冗长和更多显示的选项。 但是为此,我们首先必须看看JUnit扩展模型的另一个Struts。

自定义注释

JUnit 5 API由注释驱动,并且引擎在检查它们是否存在时会做一些额外的工作:它不仅在类,方法和参数上查找注释,还在其他注释上查找。 并且它将发现的所有内容都视为立即存在于检查的元素上。 注释可以通过所谓的meta-annotations进行注释 ,很酷的是,所有JUnit注释都是完全meta的。

这样就可以轻松地创建和编写在JUnit 5中完全可用的注释:

/*** We define a custom annotation that:* - stands in for '@Test' so that the method gets executed* - has the tag "integration" so we can filter by that,*   e.g. when running tests from the command line*/
@Target({ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Test
@Tag("integration")
public @interface IntegrationTest { }

然后我们可以像这样使用它:

@IntegrationTest
void runsWithCustomAnnotation() {// this gets executed// even though `@IntegrationTest` is not defined by JUnit
}

或者我们可以为扩展创建更简洁的注释:

@Target({ ElementType.TYPE, ElementType.METHOD, ElementType.ANNOTATION_TYPE })
@Retention(RetentionPolicy.RUNTIME)
@ExtendWith(ExternalDatabaseExtension.class)
public @interface Database { }

现在我们可以使用@Database代替@ExtendWith(ExternalDatabaseExtension。 )。 并且由于我们添加了ElementType ANNOTATION_TYPE到允许的目标列表中,它也是一个元注释,我们或其他人可以对其进行进一步的组合。

假设我们要对某些测试的运行时间进行基准测试。 首先,我们创建要使用的注释:

@Target({ ElementType.TYPE, ElementType.METHOD, ElementType.ANNOTATION_TYPE })
@Retention(RetentionPolicy.RUNTIME)
@ExtendWith(BenchmarkCondition.class)
public @interface Benchmark { }

它已经指向BenchmarkCondition ,我们将在接下来实现。 这是我们的计划:

  • 衡量整个测试类的运行时间,存储执行任何测试之前的时间
  • 衡量各个测试方法的运行时间,存储每次测试之前的时间
  • 执行测试方法后,检索测试的启动时间,计算并打印结果运行时
  • 执行完所有测试后,检索类的启动时间,计算并打印结果运行时
  • 仅当用@Benchmark注释类或方法时,才执行任何此操作

最后一点可能不是立即显而易见的。 为什么扩展未处理未使用@Benchmark注释的方法? 这源于以下事实:如果将扩展应用于类,它将自动应用于其中的所有方法。 因此,如果我们的要求表明我们可能希望对类进行基准测试,但不一定要对所有单个方法进行基准测试,则需要排除它们。 我们通过检查它们是否被单独注释来做到这一点。

碰巧的是,前四个点直接对应于生命周期回调BeforeAllBeforeEachAfterEachAfterAll ,因此我们要做的就是实现四个对应的接口。 这些实现非常简单,它们只是按照我们上面所说的去做:

public class BenchmarkCondition implementsBeforeAllExtensionPoint, BeforeEachExtensionPoint,AfterEachExtensionPoint, AfterAllExtensionPoint {private static final Namespace NAMESPACE =Namespace.of("BenchmarkCondition");@Overridepublic void beforeAll(ContainerExtensionContext context) {if (!shouldBeBenchmarked(context))return;writeCurrentTime(context, LaunchTimeKey.CLASS);}@Overridepublic void beforeEach(TestExtensionContext context) {if (!shouldBeBenchmarked(context))return;writeCurrentTime(context, LaunchTimeKey.TEST);}@Overridepublic void afterEach(TestExtensionContext context) {if (!shouldBeBenchmarked(context))return;long launchTime = loadLaunchTime(context, LaunchTimeKey.TEST);long runtime = currentTimeMillis() - launchTime;print("Test", context.getDisplayName(), runtime);}@Overridepublic void afterAll(ContainerExtensionContext context) {if (!shouldBeBenchmarked(context))return;long launchTime = loadLaunchTime(context, LaunchTimeKey.CLASS);long runtime = currentTimeMillis() - launchTime;print("Test container", context.getDisplayName(), runtime);}private static boolean shouldBeBenchmarked(ExtensionContext context) {return context.getElement().isAnnotationPresent(Benchmark.class);}private static void writeCurrentTime(ExtensionContext context, LaunchTimeKey key) {context.getStore(NAMESPACE).put(key, currentTimeMillis());}private static long loadLaunchTime(ExtensionContext context, LaunchTimeKey key) {return (Long) context.getStore(NAMESPACE).remove(key);}private static void print(String unit, String displayName, long runtime) {System.out.printf("%s '%s' took %d ms.%n", unit, displayName, runtime);}private enum LaunchTimeKey {CLASS, TEST}
}

有趣的细节是shouldBeBenchmarked ,它使用JUnit的API毫不费力地确定当前元素是否使用@Benchmark进行 (元注释),以及writeCurrentTime / loadLaunchTime ,后者使用存储来写入和读取启动时间。

  • 您可以在GitHub上找到代码 。

接下来的文章将讨论条件测试执行和参数注入,并显示如何使用相应扩展点的示例。 如果您迫不及待,请查看这篇文章 ,其中展示了如何将两个JUnit 4规则(条件禁用和临时文件夹)移植到JUnit 5。

摘要

我们已经看到,JUnit 4的运行者和规则对于创建干净,强大且可组合的扩展并不是理想的选择。 JUnit 5旨在通过更通用的扩展点概念来克服它们的局限性。 它们允许扩展程序指定要在测试生命周期中的哪些时间点进行干预。 我们还研究了元注释如何使自定义注释的创建变得容易。

你怎么看?

翻译自: https://www.javacodegeeks.com/2016/04/junit-5-extension-model.html

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

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

相关文章

软件工程实验5

SA17225400 哪来的妖精 《软件工程(C编码实践篇)》MOOC课程作业http://mooc.study.163.com/course/USTC-1000002006 GitHub :https://github.com/littlewulei/Software-Engineering-Lab.git 实验要求(参照视频中的具体实验过程&…

纯CSS实现水波纹效果

首先我们从结构和样式两个方面来讲解以上动图的实现过程&#xff1a;Html结构&#xff1a;<div class"square"><span></span><span></span><span></span><div class"content"><h2>Post Title</h…

乡村医生需要什么,看这张图就够了!

乡村医生需要什么&#xff0c;看这张图就够了&#xff01; 笔者最近在重庆市人民政府公开信箱中看到了一位赤脚医生写给政府的公开信&#xff0c;因读后无比感动&#xff0c;索性就摘录了出来&#xff1a; 来信内容&#xff1a; 我们是70-80年代的赤脚医生&#xff0c;是计划生…

Handsfree.js — 一个通过计算机视觉集成手势,面部表情和各种姿势识别的前端库

当电视上出现上图这种科技大片的时候&#xff0c;有没有幻想过有一天可以实现上图的这种交互&#xff0c;当我打开Handsfree这个库的介绍页时&#xff0c;看到前端页面竟然能够识别人的手势&#xff0c;面部以及各种肢体动作&#xff0c;简直刷新了我对前端能力的认知。确信这种…

ejb jsf jpa_完整的WebApplication JSF EJB JPA JAAS –第1部分

ejb jsf jpa这篇文章将是迄今为止我博客中最大的一篇文章&#xff01; 我们将看到完整的Web应用程序。 最新的技术&#xff08;直到今天&#xff09;都将完成&#xff0c;但是我将给出一些提示以显示如何使本文章适应较旧的技术。 在本文的结尾&#xff0c;您将找到要下载的源代…

Jzoj4782 Math

若一个数x是平方数&#xff0c;则d(x)为平方数 所以就是要考虑有多少对i*j为平方数 我们假设&#xff0c;ip*k^2&#xff0c;那么&#xff0c;jp*q^2时&#xff0c;i*j为平方数&#xff08;p不含平方因子&#xff0c;k&#xff0c;q为正整数&#xff09; 所以&#xff0c;我们对…

前端暗黑模式,你了解多少

关于使用越来越多的前端暗黑模式&#xff0c;手机的app或网站都将支持暗黑模式逐渐成为一种规范&#xff0c;这样做的目的是什么呢&#xff1f;从我最初的理解是为了在黑暗的环境下屏幕上阅读的体验考虑&#xff0c;但是看了文摘却有另一种意义。暗黑模式究竟能不能起到省电的作…

两全其美的

使用抽象文档模式的类型安全视图 您如何组织对象&#xff1f; 在本文中&#xff0c;我将介绍一种模式&#xff0c;该模式以无类型的方式在您的系统中组织所谓的名词类&#xff0c;然后使用特征公开数据的类型化视图。 这使得只需少量的牺牲就可以在Java之类的语言中获得JavaScr…

Windows内核函数

字符串处理 在驱动中一般使用的是ANSI字符串和宽字节字符串&#xff0c;在驱动中我们仍然可以使用C中提供的字符串操作函数&#xff0c;但是在DDK中不提倡这样做&#xff0c;由于C函数容易导致缓冲区溢出漏洞&#xff0c;针对字符串的操作它提供了一组函数分别用来处理ANSI字符…

前端应该关注的2021年UI设计趋势

UI设计趋势几乎每年都在发生变化&#xff0c;变化的原因是人们的审美在变导致的&#xff0c;还是设计越来越人性化。市场上是谁在主导设计趋势&#xff1f;其中原因不得而知&#xff0c;我们先看看究竟有哪些变化&#xff1a;1. 3D插图&#xff08;依然流行&#xff09;3D图像将…

如何让你在开发者工具中查看源代码有语法高亮和暗黑主题的效果

如何让你在Chrome浏览器开发者工具中查看源代码的时候&#xff0c;和在代码编辑器中有同样的代码语法高亮的效果&#xff0c;而且还是深色主题&#xff0c;如果你是深色主题的爱好者就更合你意了。国外的美女开发者为你实现了这样功能的浏览器拓展&#xff0c;她的Github主页&a…

“太空语言”JavaScript编码标准规范指南

喷气推进实验室是 美国国家航空航天局的科研机构。 该实验室JPL开发大部分的软件是用在无人深度太空和其他行星探测的领域。他们拥有著名的 好奇号火星探测器 和 旅行者号探测器 。已经离开太阳系25年&#xff0c;仍然在飞行并提供科学信息。高水平的自动化和长期的任务导致了对…

如何在JUnit 5中替换规则

最近发布的JUnit 5&#xff08;又名JUnit Lambda&#xff09; Alpha版本引起了我的兴趣&#xff0c;在浏览文档时&#xff0c;我注意到规则以及跑步者和阶级规则都消失了。 根据文档&#xff0c;这些部分竞争的概念已被单个一致的扩展模型取代。 多年来&#xff0c; Frank和我…

微页面设计开发指南

一、目标实现左侧&#xff1a;为可用的组件列表&#xff0c;可拖动任一组件到中间的预览区域中间&#xff1a;为页面预览效果页面&#xff0c;选中任一组件&#xff0c;可在右侧进行参数配置右侧&#xff1a;为组件的参数配置&#xff08;选中中间的组件时出现&#xff09;&…

商城商品购买数量增减的完美JS效果

商城商品购买数量增减的完美JS效果 近期在开发一个地方O2O租书项目&#xff0c;使用ASP.NET MVC技术&#xff0c;其中在图书详情页&#xff0c;用户可以输入借阅的数量&#xff0c;这里使用了js来控制数量的增减和校验。 数量一定是数字 点击增减按钮的时候要能自动加1或减1 …

这款插件让你在VSCode上也能答题背单词

在VSCode上也可以在线答题了&#xff0c;插件市场上线了一款答题的插件&#xff0c;免去了去其它网站或者软件的烦恼&#xff0c;代码写累了&#xff0c;随手打开答题功能&#xff0c;换换脑子&#xff0c;或者熟悉两个单词&#xff0c;程序员的别样休闲时光&#xff0c;哈哈&a…

使用Java 8在地图上流式传输

在本文中&#xff0c;我将向您展示如何在标准Java映射上有效地实现Speedment Open Source流&#xff0c;并将Stream接口扩展为MapStream&#xff01; 即使在复杂的情况下&#xff0c;此添加将使保持流的具体性和可读性变得更加容易。 希望这将允许您继续流式传输而不会过早收集…

如何使用python给PDF文件加水印

Python作为编程界最火的语言&#xff0c;能做的事几乎你能想到的它都能干&#xff0c;就连抢茅台都可以&#xff0c;还有什么不行&#xff1f;&#xff01;Python作为脚本编程语言&#xff0c;可以做很多事情。使用Python&#xff0c;你可以轻松地给pdf加上水印。 你可以使用名…

搭建一个redis高可用系统

一、单个实例 当系统中只有一台redis运行时&#xff0c;一旦该redis挂了&#xff0c;会导致整个系统无法运行。 单个实例二、备份 由于单台redis出现单点故障&#xff0c;就会导致整个系统不可用&#xff0c;所以想到的办法自然就是备份&#xff08;一般工业界认为比较安全的备…

SSH连接远程服务器,本地known_hosts文件记录了什么

今天工作时&#xff0c;使用ssh命令远程连接公司的本地服务器时&#xff0c;突然出现以下错误bash-3.2$ ssh argus192.168.200.8 WARNING: REMOTE HOST IDENTIFICATION HAS CHANGED! IT IS POSSIBLE THAT SOMEONE IS DOING SOMETHING NASTY! Someone could be eavesdroppin…