jUnit:规则

规则在测试,测试用例或测试套件周围增加了特殊处理。 他们可以对类中的所有测试执行通用的其他验证,并发运行多个测试实例,在每个测试或测试用例之前设置资源,然后将其拆除。

该规则可以完全控制将要应用到的测试方法,测试用例或测试套件。 完全控制意味着规则决定运行它之前和之后要做什么以及如何处理引发的异常。

第一章介绍了如何使用规则,第二章介绍了内置规则可以做什么。 第三章介绍了我发现的第三方规则库,最后一章介绍了如何创建新规则。

使用规则

本章说明如何在测试用例中声明和使用规则。 大多数规则可以分别应用于每种测试方法,一次应用于整个测试用例,一次应用于整个测试套件。 为每个测试单独运行的规则称为测试规则,应用于整个测试用例或套件的规则称为类规则。

我们将以临时文件夹规则为例,因此第一个子章节将说明它的作用。 第二小节将其声明为测试规则,第三小节将其声明为类规则。 最后一个子章节显示了如何从测试内部访问该文件夹。

规则示例–临时文件夹

临时文件夹规则创建一个新的空文件夹,运行测试或测试用例,然后删除该文件夹。 您可以指定在哪里创建新文件夹,也可以在系统临时文件目录中创建它。

临时文件夹既可以用作测试规则,也可以用作类规则。

声明测试规则

测试规则(例如,针对每种测试方法分别运行的规则)必须在带有@Rule注释的公共字段中声明。

声明测试规则:

public class SomeTestCase {@Rulepublic TemporaryFolder folder = new TemporaryFolder();
}

上面的folder规则会在每种测试方法之前创建一个新文件夹,然后将其销毁。 所有测试都可以使用该目录,但不能通过该目录共享文件。 由于我们使用的构造函数没有参数,因此该文件夹将在系统临时文件目录中创建。

测试规则在使用@Before注释的方法之前和之后使用@After注释的方法之前进行工作。 因此,他们也将有权访问临时文件夹。

宣布班级规则

类规则(例如,针对整个测试用例或测试套件运行一次的规则)必须在公共静态字段中声明,并使用@ClassRule注释进行注释。

声明测试用例规则:

public class SomeTestCase {@ClassRulepublic static TemporaryFolder folder = new TemporaryFolder();
}

上面的folder规则在运行第一个测试方法之前创建一个新文件夹,并在最后一个测试方法之后销毁它。 所有测试都可以使用该目录,并且可以查看以前运行的测试所创建的文件。

类规则在该类内部的任何内容之前运行。 例如,用@BeforeClass@AfterClass注释的方法也可以访问临时文件夹。 该规则在它们之前和之后运行。

在测试中使用规则

规则与其他任何规则一样都是类,测试可以自由调用其公共方法并使用其公共字段。 这些调用用于将测试特定的配置添加到规则或从中读取数据。

例如,可以使用newFilenewFoldergetRoot方法访问临时文件夹。 前两个在临时文件夹中创建新文件或文件夹,而getRoot方法返回临时文件夹本身。

创建临时文件和文件夹:

@Test
public void test1() {// Create new folder inside temporary directory. Depending on how you // declared the folder rule, the directory will be deleted either // right after this test or when the last test in test case finishes.File file = folder.newFolder("folder");
}@Test
public void test2() {// Create new file inside temporary folder. Depending on how you // declared the folder rule, the file will be deleted either // right after this test or when the last test in test case finishes.File file = folder.newFile("file.png");
}

默认规则

JUnit带有五个直接可用的规则 :临时文件夹,预期的异常,超时,错误收集器和测试名称。 临时文件夹已在上一章中进行了说明,因此我们将仅简要介绍其余四个规则。

预期异常

预期异常将运行测试并捕获其引发的所有异常。 该规则能够检查异常是否包含正确的消息,正确的原因以及是否由正确的行抛出。

预期的异常具有私有构造函数,必须使用静态none方法进行初始化。 每个异常引发测试都必须配置预期的异常参数,然后调用规则的expect方法。 该规则在以下情况下失败:

  • 测试在expect方法调用之前引发任何异常,
  • expect方法调用之后,测试不会引发异常,
  • 引发的异常没有正确的消息,类或原因。

最后一条测试行引发异常。 在导致异常之前立即配置了预期的异常规则:

@Rule
public ExpectedException thrown= ExpectedException.none();@Test
public void testException() {// Any exception thrown here causes failuredoTheStuff();// From now on, the rule expects NullPointerException exception// to be thrown. If the test finishes without exception or if it // throws wrong one, the rule will fail.thrown.expect(NullPointerException.class);// We well check also messagethrown.expectMessage("Expected Message.");// this line is supposed to throw exceptiontheCodeThatThrowsTheException();
}

奖励:期望的消息方法也接受hamcrest匹配器参数。 这样,您就可以测试消息前缀后缀,无论它是否与某些正则表达式匹配或与其他任何正则表达式匹配。

超时

超时规则可以同时用作测试规则和类规则。 如果将其声明为测试规则,则它将相同的超时限制应用于该类中的每个测试。 如果将其声明为类规则,则它将超时限制应用于整个测试用例或测试套件。

错误收集器

通过错误收集器,您可以在测试内部运行多个检查,然后在测试结束后立即报告所有失败。

期望值与实际值的断言使用规则公开的checkThat方法进行评估。 它接受hamcrest匹配器作为参数,因此可以用来检查任何内容。

可以使用addError(Throwable error)方法直接报告意外异常。 或者,如果有要运行的Callable实例,则可以通过checkSucceeds方法调用它, checkSucceeds方法将所有抛出的异常添加到错误列表中。

测试名称

测试名称规则在测试内部公开测试名称。 当您需要创建自定义错误报告时,它可能会很有用。

第三方规则库

规则与测试类分离,因此很容易编写通用规则库并在项目之间共享它们。 本章介绍了三个这样的库。

系统规则是用于测试使用java.lang.System的代码的规则集合。 它有充分的文档证明,可在maven中使用,并根据Common Public License 1.0(与jUnit相同)发布。 系统规则使您可以轻松:

  • 测试System.errSystem.out内容,
  • 模拟System.in输入,
  • 配置系统属性并将其值还原回
  • 测试System.exit()调用–是否被调用以及返回的值是什么,
  • 自定义Java SecurityManager并将其还原。

一个大集的有用的规则是可以在GitHub上aisrael帐户。 它的文档有所限制,但是您可以随时查看代码 。 所有规则均根据MIT许可发布:

  • 启动和停止内存中的derby数据库 ,
  • 启动和停止默认的Java HttpServer ,
  • 启动和停止Jetty服务器,
  • 运行存根jndi ,
  • 对dbUnit测试的一些支持。

github上另一套未公开的规则。 我不会在这里列出它们,因为它们的名称是不言自明的,并且它们没有指定的许可证。 查看规则目录以查看其列表。

自订规则

本章介绍如何创建新规则。 可以通过实现TestRule接口或扩展TestRule提供的两个便捷类ExternalResourceVerifier之一来从头实现它们。

我们将从头开始创建一个新规则,然后使用ExternalResource类重写它。

新规则

新规则可确保在每个测试完成工作后,正确删除由测试创建的所有文件。 测试本身仅具有一项责任:使用规则公开的ensureRemoval(file)方法报告所有新文件。

如何声明和使用DeleteFilesRule规则:

@Rule
public DeleteFilesRule toDelete = new DeleteFilesRule();@Test
public void example() throws IOException {// output.css will be deleted whether the test passes, fails or throws an exceptiontoDelete.ensureRemoval("output.css");// the compiler is configured to create output.css filecompileFile("input.less");checkCorrectess("output.css");
}

从头开始

每个规则(包括类规则)必须实现@TestRule接口。 该接口只有一种方法:

public interface TestRule {Statement apply(Statement base, Description description);
}

我们的工作是获取base参数中提供的语句,然后将其转换为另一个语句。 该语句表示一组要运行的动作,例如测试,测试用例或测试套件。 它可能已经被其他声明的规则修改,并且包括在测试或类方法之前和之后。

第二个description参数描述输入语句。 它可以告诉测试类名称,测试名称,放置在其上的注释,它知道我们是在处理测试还是在测试套件等。我们将不需要它。

我们需要创建一个新的语句,它将执行三件事:

  • 清空要删除的文件列表。
  • 运行底层测试,测试案例或测试套件由所表示的base参数。
  • 删除先前运行的语句中测试报告的所有文件。

该语句是具有一个抽象方法的类:

public abstract class Statement {public abstract void evaluate() throws Throwable;
}

由于基础语句可能会引发异常,因此删除所有文件的代码必须在finally块中运行:

public class DeleteFilesRule implements TestRule  {public Statement apply(final Statement base, final Description description) {return new Statement() {@Overridepublic void evaluate() throws Throwable {emptyFilesList(); // clean the list of filestry {base.evaluate(); // run underlying statement} finally {removeAll(); // delete all new files}}};}
}

引用的两个方法emptyFilesListremoveAll都在new语句外部,直接在DeleteFilesRule类内部声明:

public class DeleteFilesRule implements TestRule  {private List<File> toDelete;private void emptyFilesList() {toDelete = new ArrayList<File>();}private void removeAll() {for (File file : toDelete) {if (file.exists())file.delete();}}/* ... the apply method ... */
}

我们需要的最后一件事是能够添加要删除的文件的公共方法:

public void ensureRemoval(String... filenames) {for (String filename : filenames) {toDelete.add(new File(filename));}
}

全班

public class DeleteFilesRule implements TestRule  {private List<File> toDelete;public void ensureRemoval(String... filenames) {for (String filename : filenames) {toDelete.add(new File(filename));}}private void emptyFilesList() {toDelete = new ArrayList<File>();}private void removeAll() {for (File file : toDelete) {if (file.exists())file.delete();}}public Statement apply(final Statement base, final Description description) {return new Statement() {@Overridepublic void evaluate() throws Throwable {emptyFilesList(); // clean the list of filestry {base.evaluate(); // run underlying statement} finally {removeAll(); // delete all new files}}};}
}

扩展内置类

JUnit包含两个便捷类ExternalResourceVerifier旨在进一步简化上述过程。

外部资源

当您需要围绕基础测试语句进行某种预处理和后处理时, ExternalResource帮助。 如果需要预处理,请覆盖before方法。 如果需要后处理,请覆盖after方法。 的
从finally块调用after ,无论如何它将运行。

我们的DeleteFilesRule可以这样重写:

public class DeleteFilesRule2 extends ExternalResource  {/* ... list, ensureRemoval and removeAll methods ... */@Overrideprotected void before() throws Throwable {toDelete = new ArrayList<File>();}@Overrideprotected void after() {removeAll();}}

验证者

Verifier者只有一种verify要覆盖的方法。 只有在包装测试未引发异常后,该方法才会运行。 顾名思义,如果您想在测试后进行其他检查,则验证器是不错的选择。

有关jUnit的更多信息

关于jUnit 4功能的上一篇文章:

  • jUnit:动态测试生成

翻译自: https://www.javacodegeeks.com/2014/09/junit-rules-2.html

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

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

相关文章

常用浏览器内核:

浏览器内核又可以分为两部分&#xff1a;渲染引擎和JS引擎。 PC端&#xff1a;IE&#xff1a;Trident&#xff0c;沿用到IE11,即兼容模式。 IE8 的 JavaScript 引擎是 Jscript&#xff0c;IE9&#xff08;PS: JS内核&#xff09; 开始用 Chakra&#xff0c;这两个版本区别很大…

行内格式化

相对于熟知的块级格式化上下文&#xff0c;行内格式化上下文更加的复杂难明。行内元素不像块级元素那样直来直去&#xff0c;一个块级元素占据一行&#xff0c;其他块级元素在垂直方向依次向下排列即可。行内元素不同&#xff0c;多个行内元素可以在一行显示&#xff0c;那么&a…

[转载]struts+hibernate遇到的错误总结

原文地址&#xff1a;strutshibernate遇到的错误总结作者&#xff1a;畫上句號经过对strutshibernate几天的学习&#xff0c;大体上还算比较的了解机制&#xff0c;以前学习的时候都是 单个框架训练&#xff0c;没有结合2个框架做&#xff0c;所以今天就找了个网上发布租房信息…

JPA休眠替代方案。 如果JPA或Hibernate对于我的项目而言不够好,该怎么办?

你好&#xff01;你好吗&#xff1f; 今天&#xff0c;我们将讨论不建议使用JPA / Hibernate的情况。 在JPA领域之外&#xff0c;我们还有哪些选择&#xff1f; 我们将谈论的是&#xff1a; JPA /休眠问题 解决一些JPA /休眠问题的方法 选择此处描述的框架的标准 Spring J…

一个Web前端自学者的自述

想来想去还是写下这篇文章&#xff0c;先说明&#xff0c;我精通JAVA编程语言和web前端常见的技术&#xff0c;个人是做JAVA的多&#xff0c;但是更加喜欢前端。因为我从高一开始接触JAVA&#xff0c;家父是黑马的JAVA讲师&#xff0c;自己对编程很热爱&#xff0c;在大学的时候…

mongoose中的populate之多级填充,嵌套字段填充?

在mongoose中存引用的时候如果是多级&#xff0c;查询的时候填充引用字段会使用populate&#xff0c;如下&#xff1a; 定义一个User&#xff0c;有字段friends每一项是自己collection的ObjectId。 // file: user-schema.js let mongoose require(mongoose) let ObjectId m…

深度学习优化基础

1、网络优化参数 sigmoid函数&#xff1a;1/(1e^(-x))&#xff1a;&#xff1a;便于求导的平滑函数&#xff0c;但是容易出现梯度消失问题&#xff1b;函数中值不是0&#xff0c;会导致模型训练的收敛速度变慢。。。 tanh函数&#xff1a;(e^x-e^(-x))/(e^xe^(-x))::解决了zero…

稳定高效大型系统架构---集群中间件开发

那现在来说&#xff0c;稳定的中间件应该是什么样子呢&#xff1f; 对于客户端请求&#xff0c;如果发现服务停止&#xff0c;可以实现服务无缝转移&#xff0d;&#xff0d;&#xff0d;这叫不丢失任何服务. 对于多个客户端请求&#xff0c;可以讲请求轮巡到不同的服务器上&am…

css实现web前端最美的loading加载动画!

这些好看的loading效果&#xff0c;你还只会用第三方库吗&#xff1f;CSS3教你实现 ​前言 loading效果在实际开发中是很常见的&#xff0c;尤其是在Ajax请求的时候&#xff0c;可以给用户一个很好的交互体验。 今天这篇文章我们一起来看看如何通过CSS3实现各种不同的loadin…

如何使用Hibernate从Play生成DDL脚本! 框架项目

好的&#xff0c;因此您一直在使用hibernate属性名称“ hibernate.hbm2ddl.auto ” value “ 更新 ”来不断更新数据库架构&#xff0c; 但是现在您需要一个完整的DDL脚本吗&#xff1f; 从您的Global Class onStart中使用此方法来导出DDL脚本。 只需为其提供实体的包名称&…

C# 设计模式,工厂方法

C#工厂方法 1 using System;2 using System.Collections.Generic;3 using System.Linq;4 using System.Text;5 using System.Threading.Tasks;6 7 namespace 工厂方法 {8 class Program {9 static void Main(string[] args) { 10 IFacotry i new Fact…

javascript中令人迷惑的this

JS中的this很多时候会让人捉摸不透&#xff0c;不知道这个this到底指向的是什么。现在根据自己的理解写下这篇文章做一个总结。 我们知道this指向谁一般情况下是在运行时决定的&#xff0c;并且运行时决定this指向的因素又有很多&#xff0c;例如是不是被bind了&#xff0c;或…

容易忽视但是功能灰常强大的Java API(五. 二分查找)

五. 二分查找 二分查找是一个高效的查找算法&#xff0c;在java的集合对象中也提供了二分查找的算法&#xff0c;如下面的java api接口&#xff1a; java.util.Arrays.binarySearch(java.lang.Object,java.lang.Object,java.util.Comparator) java.util.Arrays.binarySear…

打印 PRINT

打印 PRINT 字符串和数值类型 可以直接输出。 print(1) #out:1 print(a) #out:a 变量 无论什么类型&#xff0c;数值&#xff0c;字符串&#xff0c;列表&#xff0c;字典...都可以直接输出 n 1 s a list_a [1,3,4] dict_c {a:3,b:4} print(n) #out&…

css3帮你轻松实现圆角效果,不一样的前端页面。

在Web前端页面实现圆角效果&#xff0c;CSS3帮你轻松实现&#xff0c;一个人人皆知的属性 圆角边框的绘制是Web页面和Web应用程序中经常用来美化页面效果的手法之一。今天&#xff0c;小编为大家介绍CSS3提供的可以将矩形变为圆角矩形的一个属性 技术等级&#xff1a;中级 | 适…

使用Vysper,TomEE和PrimeFaces将XMPP服务器嵌入JSF Web应用程序内部

我有一个需要在完成某些工作时通知用户的应用程序。 它使用JSF和Primefaces&#xff0c;因此可以使用大气 &#xff08;也称为Push&#xff09;来实现这种通知。 但是另一个有趣的方法是使用嵌入在Java Web应用程序中的XMPP服务器。 好的&#xff0c;好的&#xff0c;您不必嵌…

appium和selenium不同与相同之处

原文来自&#xff1a; https://www.cnblogs.com/zhengshuheng/p/6370398.html selenium是web端的自动化&#xff0c;appium是app端的自动化&#xff0c;它继承了webdriver(也就是selenium 2) 转载于:https://www.cnblogs.com/lv-lxz/p/11118862.html

Mockito 101

Mockito是一个模拟框架&#xff0c;可让您使用简洁的API编写漂亮的测试。 它偏向于最低规格&#xff0c;使不同的行为看起来有所不同&#xff0c;并显示清晰的错误消息。 创造嘲弄 要使用Mockito创建模拟&#xff0c;只需使用Mock注释模拟&#xff0c;然后调用MockitoAnnotati…

前端开发常用代码片段(下篇)

二十二、正则表达式 //验证邮箱/^\w ([0-9a-zA-Z] [.]) [a-z]{2,4}$///验证手机号/^1[3|5|8|7]\d{9}$///验证URL/^http:\/\/. \.///验证身份证号码/(^\d{15}$)|(^\d{17}([0-9]|X|x)$)///匹配字母、数字、中文字符/^([A-Za-z0-9]|[\u4e00-\u9fa5])*$///匹配中文字符/[\u4e00-\u9…

使用 Visual Studio 编译 wget 为库文件

添加代码与预编译指令与上一篇使用 Visual Studio 编译 wget 为可执行文件一致&#xff0c;区别在于这回建的是静态库工程&#xff08;编译为动态库过程类似:)&#xff09; 从wget的main函数开始读下来&#xff0c;发现问题不少&#xff0c;程序可能基于效率或者编码方便的因素…