junit4 集成测试_使用JUnit规则进行干净的集成测试

junit4 集成测试

JUnit Rules的优势,尤其是在进行集成测试时,几乎不能被高估。 在本文中,我们将阐明ExternalResource扩展的有用性。 在我们必须使用抽象外部资源的第三方库的情况下,这简化了灯具控制。 作为示例,我们将看看如何基于Git提交日志消息来验证对条目列表的正确检索。

什么是集成测试?

“关注点分离”可能是软件设计和实现中最重要的单个概念。
语用单元测试[HUTH03]

通常,我们使用单元测试来检查一小段生产代码是否按预期工作。 但是重要的是要了解,这类测试仅限于开发人员负责的代码。 为了澄清这一点,请考虑合并第三方库来管理对文件,数据库,Web服务等的访问。

测试将隐式调用第三方组件的代码,因为我们的被测系统 (SUT)依赖于这些组件(DOC)[MESZ07]。 如果外部资源之一不可用,尽管开发人员的代码可能没有问题,但它们将失败。 此外,访问这些资源通常很慢,并且设置测试装置通常很麻烦。 更不用说脆弱性了,它是由不同库版本的潜在语义变化引起的。

所有这些缺点建议通过适配器抽象[FRPR10]将应用程序的代码与第三方代码分开。 不仅仅是抽象适配器组件可以在应用程序的问题域方面提供表达性的API,它还允许使用轻量级的替代测试double (通常表示为嘲笑)来替换基于第三方代码的实现。

使用JUnit进行测试

使用Junit书本封面进行测试 使用JUnit进行测试是Java开发人员可以学习的最有价值的技能之一。 无论您的背景是什么,无论您是只是想建立一个安全网以减少桌面应用程序的性能下降,还是要基于健壮且可重复使用的组件来提高服务器端的可靠性,都需要进行单元测试。

弗兰克(Frank)写了一本书,它为使用JUnit进行测试的基本知识提供了深刻的切入点,并为您准备与测试相关的日常工作挑战做好了准备。

学到更多…

这消除了先前列出的有关单元测试的依赖性问题。 双重测试的设置成本低廉,可以将测试中的系统与第三方代码隔离开,并保持测试快速可靠[MESZ07]。 但是,它剩下的工作是测试适配器组件的正确行为。 这是集成测试开始起作用的时候。

该术语指的是软件测试中的阶段,在该阶段中,将各个软件模块组合在一起并作为一个组进行测试[INTTES]。 可以说,我们使用适配器抽象将一个或多个第三方模块组合在一起以提供某种功能。 由于从应用程序的角度来看,此类适配器是低级组件,因此该策略隐式导致了一种自下而上的方法,该方法首先测试最低级别的组件,然后再用于促进更高级别组件的测试。

您可能想知道为测试目的调整设计是否不是一件坏事。 但是,通过使用适配器,您可以确定应用程序和第三方代码之间的界限。 如果新的库版本引入了稍微不同的行为,则只需调整适配器代码即可再次通过相应的集成测试。 您的实际应用程序代码(包括单元测试)将不受影响! 此外,您可以通过提供适当的适配器轻松切换到其他供应商。 因此,遵循这种做法还可以带来更健康的应用程序设计。 [APPE15]

外部资源处理

不幸的是,在编写集成测试时,我们必须面对通过使用测试倍数来避免单元测试所遇到的问题。 特别是从编码角度来看,安装测试夹具通常需要大量的精力。 最重要的是,我们还必须做好适当的内务处理[MESZ07]。 例如,这意味着我们可能需要在测试执行后重置外部资源的状态。 后者对于确保后续测试独立运行很重要。 这样一来,测试所做的资源修改就不会伪造其后继者的验证结果。

为了减少设置和拆卸代码的经常性开销,将普通段落交换到测试帮助程序类中似乎很自然。 考虑一下系统环境变量,主数据记录等的创建,删除或操作。 JUnit规则是特殊的测试助手,它像AOP框架一样拦截测试方法调用。 与AspectJ中的环境建议相比,它们可以在实际测试执行之前和/或之后做有用的事情。 例如,可以在测试运行之前注册REST服务资源,并在测试结束后自动将其删除。

JUnit为规则提供了方便的基类ExternalResource ,这些规则用于在测试之前设置外部资源(文件,套接字,服务器,数据库连接等),并保证随后将其拆除[EXRAPI]。 以下清单ServerRule显示了原理。

public class ServerRule extends ExternalResource {private final int port;public ServerRule( int port ) {this.port = port;}@Overrideprotected void before() throws Throwable {System.out.println( "start server on port: " + port );}@Overrideprotected void after() {System.out.println( "stop server on port: " + port );}
}

ServerRule的构造函数使用我们的虚拟服务器类型的端口号。 为了演示这一概念,我们实际上不是开始一个真正的概念,而只是打印出一个包含此数字的数字,该数字包含有关beforeafter回调挂钩的调用的消息。 下一个清单显示ServerRule的用法。

public class MyServerITest {@Rulepublic final ServerRule serverRule = new ServerRule( 5050 );@Testpublic void foo() {System.out.println( "code that fails without server access" ); }
}

请注意,该规则如何由带有@Rule注释的公共非静态字段@Rule 。 运行测试用例将导致以下输出。

start server on port: 5050
code that fails without server access
stop server on port: 5050

如您所见,该规则确保测试代码在预期的环境先决条件内执行,并自动进行内部管理。 为了加深这个主题,让我们看一个更详细的示例,该示例说明了规则管理的灯具与被测组件之间的相互作用。

为Git集成测试设计规则

标题图像显示了一个时间轴组件,该组件通过可配置的ItemProvider适配器检索其Item列表。 捕获图片时使用的适配器类型从Git存储库读取条目。 每个项目代表当前存储库分支的提交。 该插图基于我为《 Testing with JUnit》一书开发的示例应用程序的屏幕截图。 因为它超出了本书的范围,所以我借此机会对我申请编写JGit集成测试的GitRule助手进行了解释。

驱动力是提供一个实用程序类,其目的是简化建立包含任意提交,分支等内容的git夹具存储库的任务。 为此,我创建了一个GitRepository类型。 这通过JGit处理本地存储库上的存储库交互。 以下摘录应阐明概念。

public class GitRepository {private final File location;GitRepository( File location ) {this.location = location;}public RevCommit commitFi1e( String fileName, String content, String message )throws IOException{createFi1e( fileName, content );addFi1es();return commit( message );}[...]
}

如您所见, GitRepository实例采用一个构造函数参数,该参数引用本地Git仓库的工作目录。 但是请注意构造函数的可见性限制。 这是因为抽象不负责处理存储库资源的生命周期。 对于后者,我们使用ExternalResource派生,如下面的清单所示。

public class GitRule extends ExternalResource {private final Set<File> repositories;public GitRule() {repositories = new HashSet<>();}@Overrideprotected void after() {repositories.forEach( repository -> delete( repository ) );}public GitRepository create( File location ) {createRepositoryOnDisk( location );GitRepository result = new GitRepository( location );repositories.add( location);return result;}private void createRepositoryOnDisk( File location ) {InitCommand init = Git.init();init.setDirectory( location );init.setBare( false );callInit( init );}private static void callInit( InitCommand init ) {try {init.call().close();} catch( GitAPIException exception ) {throw new GitOperationException( exception );}}
}

GitRule可以作为工厂存储特定测试所需的资源库的资源。 此外,一旦完成测试执行,它就会跟踪正确处置所需的位置。 所示版本仅在磁盘上创建本地存储库,但是当然可以对其进行增强以克隆远程存储库。

ItemProvider接口依赖于扩展类型Item的通用类型参数。 因此, GitItemProvider类型返回GitItem实例作为查找结果,并且每个git项目都是JGit RevCommit的封装。 这样说,很明显,第三方代码抽象可能会影响多个类。 以下代码段显示了一个简单的集成测试方案。 GitRule提供适用于创建实际提交的存储库。 后者用于验证GitItem实例的正确实例化。

public class GitItemTest {@Rule public final TemporaryFolder temporaryFolder = new TemporaryFolder();@Rule public final GitRule gitRule = new GitRule();@Testpublic void ofCommit() throws IOException {GitRepository repository = gitRule.create( temporaryFolder.newFolder() );RevCommit commit = repository.commitFi1e( "file", "content", "message"  );GitItem actual = GitItem.ofCommit( commit );assertThat( actual ).hasId( getId( commit ) ).hasTimeStamp( getTimeStamp( commit ) ).hasContent(  getContent( commit ) ).hasAuthor( getAuthor( commit ) );}[...]
}

该测试使用TemporaryFolder规则来确保在可访问目录下创建存储库。 实际上,使用临时文件夹规则应该使GitRule的资源删除GitRule多余。 但是,由于它的默认清除机制不会检查资源删除是否成功(无论如何,硬检查仅适用于最新的JUnit版本),所以我选择不依赖它。 这很重要,因为使用JGit可以很容易地遇到打开文件句柄的问题。

此外,测试的验证是通过定制的定制GitItemAssert断言类和一些实用程序方法(静态导入)完成的。 有了这个适当的位置之后,我们准备看一下更复杂的场景。

public class GitItemProviderITest {private static final String CLONE_NAME = "test";private static final int INITIAL_COMMIT_COUNT = 6;@Rule public final TemporaryFolder temporaryFolder = new TemporaryFolder();@Rule public final GitRule gitRule = new GitRule();private GitRepository repository;private GitItemProvider provider;private File remoteLocation;private File destination;@Beforepublic void setUp() throws IOException {remoteLocation = temporaryFolder.newFolder();repository = createRepository( remoteLocation );destination = temporaryFolder.newFolder();provider = new GitItemProvider( remoteLocation.toURI().toString(),destination,CLONE_NAME );}@Testpublic void fetchItems() throws IOException {int fetchCount = INITIAL_COMMIT_COUNT / 3;List<GitItem> actual = provider.fetchItems( null, fetchCount );assertThat( actual ).isEqualTo( subList( 0, fetchCount ) ).hasSize( fetchCount );}private List<GitItem> subList( int fromIndex, int toIndex ) {return repository.logAll().stream().map( commit -> ofCommit( commit ) ).collect( toList() ).subList( fromIndex, toIndex );}[...]
}

设置与之前的测试相似。 但是,我们的装置存储库是通过委派给createRepository方法来创建的。 为了简洁起见,我在此省略了详细信息,因为该方法仅创建带有INITIAL_COMMIT_COUNT提交的存储库。 被测试的GitItemProvider组件采用三个构造函数参数。 第一个是夹具库的位置,它将由提供者克隆。 为此,第二个参数定义一个目标目录,而第三个参数将插入克隆存储库的文件夹名称。

在练习阶段,组件从其克隆的存储库中获取可用提交的子集。 subList验证了此列表,该列表是由我们的灯具存储库中的方法subList计算得出的。 最后,这些规则负责客房整理。

如果要查看完整的示例代码,请参阅GitHub存储库https://github.com/fappel/Testing-with-JUnit上可用的示例应用程序源。

摘要

这篇文章介绍了如何在编写集成测试时将JUnit规则用于干净的资源管理。 我们已经对什么是集成测试有了基本的了解,了解了ExternalResource测试实用程序扩展的工作原理,并详细阐述了使用示例。 当然,它的意义不仅仅在于初见。 熟悉此处显示的原理后,您可能会考虑研究其他主题,例如使用ClassRule来处理持久性固定装置, 规则链 , 环境变量等。

不能不告诉您我的书《 使用JUnit进行测试 》第6章“ 用JUnit规则减少样板”可作为免费阅读样本, 网址为https://www.packtpub.com/packtlib/book/Application%20Development/ 9781782166603/6 。 如果您还不厌倦我的麻烦,请大胆前进并抓住机会深入研究JUnit规则的世界…

因此,请记住,人们始终遵守规则–不要忘了分享知识&#55357;&#56841;

资源资源

  • [APPE15]:Appel, 使用JUnit测试 ,Packt Publishing,2015年
  • [EXRAPI]:ExternalResource,API DOC, http ://junit.org/apidocs/org/junit/rules/ExternalResource.html
  • [FRPR10]:Freeman,Pryce, 不断发展的面向对象软件,由Tests指导 ,Addison Wesley,2010年
  • [HUTH03]:Hunt,Thomas, 实用单元测试有限公司,2003年,2004年
  • [INTTES]:Wikipedia,集成测试, https ://en.wikipedia.org/wiki/Integration_testing
  • [MESZ07]:Meszaros, xUnit测试模式 ,Pearson Education,Inc.,2007年

翻译自: https://www.javacodegeeks.com/2015/09/clean-integration-testing-with-junit-rules-3.html

junit4 集成测试

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

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

相关文章

多项式在matlab中的应用,matlab的应用-多项式函数及多项式拟合

matlab的应用-多项式函数及多项式拟合 Matlab 的应用- 多项式函数及多项式拟合 本节将向大家简要介绍 matlab 在多项式处理方面的应用。 多项式函数主要有&#xff1a; roots 求多项式的根 poly 特征多项式 polyval 多 项式的计算 poly2str(p, x )多项式代换 polyfit 多项式曲线…

乐惠科技php面试题,程序员中的奇葩,使用php构建魔兽世界

这是用PHP编写的魔兽世界服务器。现在它已经调试了登录服务器的过程。目前的魔兽世界客户端是2.4.3_8606。服务器列表和帐户密码数据需要查询AUTH库。世界服务器身份验证过程已完成&#xff0c;数据包加密已完成后续进程正在开发中......数据库文件在根目录: sql/sql.7z今天在群…

javafx窗体程序_JavaFX真实世界应用程序:EIZO CuratOR Caliop

javafx窗体程序JavaFX Real-World应用程序第四号称为Caliop 。 它是EIZO为医院手术室开发的CuratOR解决方案的前端。 前端在壁挂式控制台上运行&#xff0c;并允许操作团队查找有关患者的信息&#xff0c;控制各种视频源到不同显示器的路由&#xff0c;录制视频&#xff0c;拍摄…

java8 streams_Java 8 Streams:过滤和谓词否定

java8 streams最近&#xff0c;有关LJC 邮件列表的成员在流中.filter方法中使用谓词否定的有趣讨论&#xff0c;因此我认为值得在博客文章中进行总结。 讨论是关于使用.filter和否定谓词的方法。 这篇文章的代码可以在我的github帐户中找到 。 也许这就是您可能会想到的方式&…

hystrix熔断 简介_Hystrix简介– Hello World

hystrix熔断 简介在以前的博客文章中&#xff0c;我介绍了需要像Netflix Hystrix这样的库的动机。 在这里&#xff0c;我将跳入一些非常基本的方法来开始使用Hystrix&#xff0c;并在更复杂的用例中进行跟进。 你好&#xff0c;世界 以下是“ Hystrix命令”的一个简单的Hello …

php中哪个函数用于读取文件,PHP内置函数fget读取文件

php fgets()函数从文件指针读取一行语法:fgets(file,length)参数说明必需的. 指定姚要读取的文件长度可选. 指定姚都区的字节数. 默认值为102字节从文件所指向的文件中读取一行&#xff0c;并返回最大长度为1个字节的字符串. 遇到换行符(包含在返回值中&#xff0c;)&#xff0…

c++返回指针时候注意提防_提防Java中的函数式编程!

c返回指针时候注意提防这对函数式编程并不会造成太大的影响&#xff0c;这真棒。 这是关于某些实践的警告&#xff0c;您很可能会将其应用于您的代码&#xff0c;而这是完全错误的&#xff01; 。 高阶函数对于函数式编程是必不可少的&#xff0c;因此&#xff0c;谈论它们将帮…

centos php支持yaf,CentOS环境下给PHP7.0安装yaf扩展

|CentOS环境下给PHP7.0安装yaf扩展在CentOS环境下给PHP7.0安装yaf扩展&#xff0c;首先要知道PHP的安装目录在哪里&#xff0c;以我当前的路径为例&#xff0c;在/usr/local/php目录下。下一步需要下载扩展包&#xff0c;进入http://pecl.php.net/package/yaf寻找符合版本要求的…

jsf教程_JSF范围教程– JSF / CDI会话范围

jsf教程会话作用域跨越多个HTTP请求-响应周期&#xff08;理论上是无限的&#xff09;。 当您需要每个HTTP请求-响应周期进行一次交互时&#xff0c;请求作用域在任何Web应用程序中都非常有用。 但是&#xff0c;当您需要对属于用户会话的任何HTTP请求-响应周期可见的对象时&…

linux查询设备文件信息失败怎么办,Linux下使用blkid命令查询设备及文件系统信息的方法...

在Linux下可以使用blkid命令对查询设备上所采用文件系统类型进行查询。blkid主要用来对系统的块设备(包括交换分区)所使用的文件系统类型、LABEL、UUID等信息进行查询。要使用这个命令必须安装e2fsprogs软件包。直接使用blkid可列出当前系统中所以已挂载文件系统的类型。默认情…

linux rpm mysql mysql_config,Linux CentOS6 mysql rpm安装

选择操作系统及版本&#xff1a;点击5.6版本链接将下载好的文件上传至服务器下面开始安装&#xff1a;检查是是否有mysql的包# rpm -qa | grep -i mysql删除原有包# yum -y remove mysql-libs*开始安装下载好的三个文件&#xff1a;# rpm -ivh MySQL-server-5.6.38-1.el6.x86_6…

java 可视化_可视化Java 9模块关系

java 可视化正如我在之前的文章中所述 &#xff0c;我已经在Java 9 Jigsaw构建上运行Eclipse Neon了一段时间&#xff0c;并且没有任何问题。 我在周末花了几个小时来修改一些模块化工具的想法。 我为Eclipse Neon写了一个小插件&#xff0c;可视化了各个模块之间的连接。 现…

zybo的linux开发教程,Zybo全栈开发入门教程——连载三:创建Linux设备驱动和应用程序...

作者&#xff1a;Commanderfranz&#xff0c;编译&#xff1a; kenshin通过前面两篇文章我们不仅创建的自定义IP模块还移植了Linux操作系统&#xff0c;今天这篇文章的内容是将这两部分联系起来&#xff0c;其实我们创建的myLed IP相对于Linux操作系统可以是它的一个底层设备&a…

linux 二进制差分工具,打造Android万能的软件更新库

今日科技快讯阿里星球今天在苹果商店发布的更新动态显示&#xff0c;其将在近期停止APP内的音乐服务&#xff0c;用户可以通过新版本内的指引和说明&#xff0c;导出本地音乐。这意味着上线八个月后&#xff0c;阿里星球做出了一个重大决定&#xff1a;停止音乐服务后&#xff…

lambda 分类聚合_使用Java 8 Lambda,流和聚合

lambda 分类聚合总览 在本文中&#xff0c;我们将介绍使用Java 8 lambda&#xff0c;流和聚合来过滤和处理Collection中的对象。 这篇文章中的所有代码都可以在此处的 BitBucket中找到 。 对于此示例&#xff0c;我们将创建许多对象&#xff0c;这些对象代表我们IT基础架构中的…

linux镜像文件不要大于4g,Systemback制做大于4G的Ubuntu系统镜像

1 安装Systemback依此执行以下命令。sudo apt-get updatesudo add-apt-repository ppa:nemh/systembacksudo apt-get update && sudo apt-get install systemback unionfs-fuse安装完成&#xff1a;2 使用Systemback生成镜像文件输入管理员密码&#xff0c;打开后界面以…

黑马2016java_2016年成功的Java开发人员简介

黑马2016java2015年即将结束。 现在该总结过去一年中已完成的工作和未完成的工作。 此外&#xff0c;现在是预测下一个2016年的好时机。 您已经猜到这篇文章是关于2016年理想的Java开发人员的。 我想给你一个惊喜&#xff0c;这次我更改了预测的格​​式。 为了使预测更加客观…

spring一站式开发_Spring开发人员知道的一件事

spring一站式开发在最近的&#xff08;核心&#xff09;Spring框架培训课程中&#xff0c;有人问我&#xff1a;“&#xff08;Java&#xff09;Spring开发人员应该知道的一件事是什么&#xff1f;” 这个问题使我措手不及。 是的&#xff0c;&#xff08;核心&#xff09;Spri…

jax-ws和jax-rs_带有JAX-WS和Spring的Web服务应用程序

jax-ws和jax-rs1.简介 这是一个漫长的等待&#xff0c;但是我最终要发布有关使用Spring创建第一个基于SOAP的Web服务应用程序的教程。 JAX-WS &#xff08;用于XML Web服务的Java API&#xff09;是用于以XML格式创建Web服务的一组API&#xff0c;我们最常将其称为基于SOAP的We…

eclipse开发jsf_在Eclipse上创建JSF / CDI Maven项目

eclipse开发jsf当我在研究JSF和CDI示例时&#xff0c;我认为提到创建JSF和CDI Maven项目所需的步骤会很有用。 您可以找到以下步骤。 工具类 默认情况下&#xff0c;M2E插件随附的Eclipse Luna。 因此&#xff0c;无需自己安装插件。 WildFlye8.x。 从主菜单中选择文件->…