junit单元测试断言_简而言之,JUnit:单元测试断言

junit单元测试断言

简而言之,本章涵盖了各种单元测试声明技术。 它详细说明了内置机制, Hamcrest匹配器和AssertJ断言的优缺点 。 正在进行的示例扩大了主题,并说明了如何创建和使用自定义匹配器/断言。

单元测试断言

信任但要验证
罗纳德·里根(Ronald Reagan)

后期测试结构解释了为什么单元测试通常分阶段进行。 它澄清说, 真正的测试结果验证在第三阶段进行。 但是到目前为止,我们只看到了一些简单的示例,主要使用了JUnit的内置机制。

如Hello World所示,验证基于错误类型AssertionError 。 这是编写所谓的自检测试的基础。 单元测试断言将谓词评估为truefalse 。 如果为false ,则抛出AssertionError 。 JUnit运行时捕获此错误并将测试报告为失败。

以下各节将介绍三种较流行的单元测试断言变体。

断言

JUnit的内置断言机制由类org.junit.Assert 。 它提供了两种静态方法来简化测试验证。 以下代码片段概述了可用方法模式的用法:

fail();
fail( "Houston, We've Got a Problem." );assertNull( actual );
assertNull( "Identifier must not be null.",actual );assertTrue( counter.hasNext() );
assertTrue( "Counter should have a successor.",counter.hasNext() );assertEquals( LOWER_BOUND, actual );
assertEquals( "Number should be lower bound value.", LOWER_BOUND,actual );
  1. Assert#fail()无条件地引发断言错误。 这对于标记不完整的测试或确保引发了预期的异常很有帮助(另请参见“ 测试结构”中的“预期异常”部分)。
  2. Assert#assertXXX(Object)用于验证变量的初始化状态。 为此,存在两个称为assertNull(Object)assertNotNull(Object)
  3. Assert#assertXXX(boolean)方法测试boolean参数传递的预期条件。 调用assertTrue(boolean)期望条件为true ,而assertFalse(boolean)期望相反。
  4. Assert#assertXXX(Object,Object)Assert#assertXXX(value,value)方法用于对值,对象和数组进行比较验证。 尽管结果没有区别,但通常的做法是将期望值作为第一个参数,将实际值作为第二个参数。

所有这些类型的方法都提供带有String参数的重载版本。 如果发生故障,此参数将合并到断言错误消息中。 许多人认为这有助于更清楚地指定失败原因。 其他人则认为此类消息混乱,使测试更难阅读。

乍一看,这种单元测试断言似乎很直观。 这就是为什么我在前面的章节中使用它进行入门的原因。 此外,它仍然非常流行,并且工具很好地支持故障报告。 但是,在需要更复杂的谓词的断言的表达性方面也受到一定限制。

Hamcrest

Hamcrest是一个旨在提供用于创建灵活的意图表达的API的库。 该实用程序提供了称为Matcher的可嵌套谓词。 这些允许以某种方式编写复杂的验证条件,许多开发人员认为比布尔运算符更易于阅读。

MatcherAssert类支持单元测试断言。 为此,它提供了静态的assertThat(T, Matcher )方法。 传递的第一个参数是要验证的值或对象。 第二个谓词用于评估第一个谓词。

assertThat( actual, equalTo( IN_RANGE_NUMBER ) );

如您所见,匹配器方法模仿自然语言的流程以提高可读性。 以下代码片段更加清楚了此意图。 这使用is(Matcher )方法来修饰实际的表达式。

assertThat( actual, is( equalTo( IN_RANGE_NUMBER ) ) );

MatcherAssert.assertThat(...)存在另外两个签名。 首先,有一个采用布尔参数而不是Matcher参数的变量。 它的行为与Assert.assertTrue(boolean)

第二个变体将一个附加的String传递给该方法。 这可以用来提高故障消息的表达能力:

assertThat( "Actual number must not be equals to lower bound value.", actual, is( not( equalTo( LOWER_BOUND ) ) ) );

在失败的情况下,给定验证的错误消息如下所示:

hamcrest-failure

Hamcrest带有一组有用的匹配器。 图书馆在线文档的“常见匹配项”部分中列出了最重要的部分。 但是对于特定域的问题,如果有合适的匹配器,通常可以提高单元测试断言的可读性。

因此,该库允许编写自定义匹配器。

让我们返回教程的示例来讨论该主题。 首先,我们对该场景进行调整以使其更合理。 假设NumberRangeCounter.next()返回的是RangeNumber类型,而不是简单的int值:

public class RangeNumber {private final String rangeIdentifier;private final int value;RangeNumber( String rangeIdentifier, int value  ) {this.rangeIdentifier = rangeIdentifier;this.value = value;}public String getRangeIdentifier() {return rangeIdentifier;}public int getValue() {return value;}
}

我们可以使用自定义匹配器来检查NumberRangeCounter#next()的返回值是否在计数器的定义数字范围内:

RangeNumber actual = counter.next();assertThat( actual, is( inRangeOf( LOWER_BOUND, RANGE ) ) );

适当的自定义匹配器可以扩展抽象类TypeSafeMatcher<T> 。 该基类处理null检查和类型安全。 可能的实现如下所示。 请注意如何添加工厂方法inRangeOf(int,int)以便于使用:

public class InRangeMatcher extends TypeSafeMatcher<RangeNumber> {private final int lowerBound;private final int upperBound;InRangeMatcher( int lowerBound, int range ) {this.lowerBound = lowerBound;this.upperBound = lowerBound + range;}@Overridepublic void describeTo( Description description ) {String text = format( "between <%s> and <%s>.", lowerBound, upperBound );description.appendText( text );}@Overrideprotected void describeMismatchSafely(RangeNumber item, Description description ){description.appendText( "was " ).appendValue( item.getValue() );}@Overrideprotected boolean matchesSafely( RangeNumber toMatch ) {return    lowerBound <= toMatch.getValue() && upperBound > toMatch.getValue();}public static Matcher<RangeNumber> inRangeOf( int lowerBound, int range ) {return new InRangeMatcher( lowerBound, range );}
}

对于给定的示例,工作量可能会有些夸大。 但它显示了如何使用自定义匹配器消除先前帖子中有点神奇的IN_RANGE_NUMBER常量。 除了新类型外,还强制声明语句的编译时类型安全。 这意味着例如String参数将不被接受进行验证。

下图显示了使用我们的自定义匹配器时测试结果失败的样子:

hamcrest-custom-failure

很容易看出describeTodescribeMismatchSafely的实现以哪种方式影响故障消息。 它表示期望值应该在指定的下限和(计算的)上限1之间 ,并跟在实际值之后。

有点不幸的是,JUnit扩展了其Assert类的API,以提供一组assertThat(…)方法。 这些方法实际上复制了MatcherAssert提供的API。 实际上,这些方法的实现委托给这种类型的相应方法。

尽管这可能只是个小问题,但我认为值得一提。 由于这种方法,JUnit与Hamcrest库牢固地联系在一起。 这种依赖性有时会导致问题。 特别是与其他库一起使用时,通过合并自己的hamcrest版本的副本,情况甚至更糟……

Hamcrest的单元测试主张并非没有竞争。 虽然关于每次测试一个确定每个测试 一个概念的讨论超出了本文的讨论范围,但后一种观点的支持者可能认为该库的验证声明过于嘈杂。 尤其是当一个概念需要多个断言时。

这就是为什么我必须在本章中添加另一部分!

断言

在“ 测试跑步者”中,示例片段之一使用了两个assertXXX语句。 这些验证期望的异常是IllegalArgumentException的实例并提供特定的错误消息。 该段看起来像这样:

Throwable actual = ...assertTrue( actual instanceof IllegalArgumentException );
assertEquals( EXPECTED_ERROR_MESSAGE, actual.getMessage() );

上一节教我们如何使用Hamcrest改进代码。 但是,如果您碰巧是该库的新手,您可能会想知道要使用哪个表达式。 或打字可能会感到不舒服。 无论如何,多个assertThat语句会加在一起。

AssertJ库通过为Java提供流畅的断言来努力改善这一点。 流畅的接口 API的目的是提供一种易于阅读的,富有表现力的编程风格,从而减少胶合代码并简化键入。

那么如何使用这种方法来重构上面的代码?

import static org.assertj.core.api.Assertions.assertThat;

与其他方法类似,AssertJ提供了一个实用程序类,该类提供了一组静态assertThat方法。 但是这些方法针对给定的参数类型返回特定的断言实现。 这就是所谓的语句链接的起点。

Throwable actual = ...assertThat( actual ).isInstanceOf( IllegalArgumentException.class ).hasMessage( EXPECTED_ERROR_MESSAGE );

旁观者认为可读性在某种程度上得到了扩展,但无论如何都可以用更紧凑的样式来写断言。 了解如何流畅地添加与被测特定概念相关的各种验证方面。 这种编程方法支持有效的类型输入,因为IDE的内容辅助可以提供给定值类型的可用谓词列表。

因此,您想向后世提供表现力的失败消息吗? 一种可能是使用describedAs作为链中的第一个链接来注释整个块:

Throwable actual = ...assertThat( actual ).describedAs( "Expected exception does not match specification." ).hasMessage( EXPECTED_ERROR_MESSAGE ).isInstanceOf( NullPointerException.class );

该代码段期望使用NPE,但假设在运行时抛出了IAE。 然后失败的测试运行将提供如下消息:

断言失败

也许您希望根据给定的失败原因使您的消息更加细微。 在这种情况下,您可以每个验证规范之前添加一条describedAs语句:

Throwable actual = ...assertThat( actual ).describedAs( "Message does not match specification." ).hasMessage( EXPECTED_ERROR_MESSAGE ).describedAs( "Exception type does not match specification." ).isInstanceOf( NullPointerException.class );

还有更多的AssertJ功能可供探索。 但是,要使该帖子保持在范围内,请参阅实用程序的在线文档以获取更多信息。 但是,在结束之前,让我们再次看一下范围内验证示例。 这可以通过自定义断言来解决:

public class RangeCounterAssertionextends AbstractAssert<RangeCounterAssertion, RangeCounter>
{private static final String ERR_IN_RANGE_OF = "Expected value to be between <%s> and <%s>, but was <%s>";private static final String ERR_RANGE_ID = "Expected range identifier to be <%s>, but was <%s>";public static RangeCounterAssertion assertThat( RangeCounter actual ) {return new RangeCounterAssertion( actual );}public InRangeAssertion hasRangeIdentifier( String expected ) {isNotNull();if( !actual.getRangeIdentifier().equals( expected ) ) {failWithMessage( ERR_RANGE_ID, expected, actual.getRangeIdentifier()  );}return this;}public RangeCounterAssertion isInRangeOf( int lowerBound, int range ) {isNotNull();int upperBound = lowerBound + range;if( !isInInterval( lowerBound, upperBound ) ) {int actualValue = actual.getValue();failWithMessage( ERR_IN_RANGE_OF, lowerBound, upperBound, actualValue );}return this;}private boolean isInInterval( int lowerBound, int upperBound ) {return actual.getValue() >= lowerBound && actual.getValue() < upperBound;}private RangeCounterAssertion( Integer actual ) {super( actual, RangeCounterAssertion.class );}
}

自定义断言是扩展AbstractAssert常见做法。 第一个通用参数是断言的类型本身。 流利的链接样式需要它。 第二种是断言所基于的类型。

该实现提供了两种附加的验证方法,可以按照以下示例进行链接。 因此,这些方法将返回断言实例本身。 请注意, isNotNull()的调用如何确保我们要声明的实际RangeNumber不为null

定制断言由其工厂方法assertThat(RangeNumber) 。 由于它继承了可用的基本检查,因此断言可以开箱即用地验证非常复杂的规范。

RangeNumber first = ...
RangeNumber second = ...assertThat( first ).isInRangeOf( LOWER_BOUND, RANGE ).hasRangeIdentifier( EXPECTED_RANGE_ID ).isNotSameAs( second );

为了完整RangNumberAssertion ,以下是RangNumberAssertion的实际运行方式:

断言自定义失败

不幸的是,不可能在同一测试用例中使用两种不同的断言类型和静态导入。 当然,假定这些类型遵循assertThat(...)命名约定。 为了避免这种情况,文档建议扩展实用程序类Assertions

这样的扩展可用于提供静态的assertThat方法,作为所有项目自定义断言的入口。 通过在整个项目中使用此自定义实用程序类,不会发生导入冲突。 在为所有断言提供单一入口点的部分中,可以找到详细的描述在线文档中有关定制断言的 yours + AssertJ

流利的API的另一个问题是单行链接的语句可能更难调试。 这是因为调试器可能无法在链中设置断点。 此外,可能不清楚哪个方法调用已引起异常。

但是,正如Wikipedia所说的那样,可以通过将语句分成多行来克服这些问题,如上面的示例所示。 这样,用户可以在链中设置断点,轻松地逐行浏览代码。

结论

简而言之,JUnit的这一章介绍了不同的单元测试断言方法,例如该工具的内置机制,Hamcrest匹配器和AssertJ断言。 它概述了一些优缺点,并通过本教程的持续示例对主题进行了扩展。 此外,还展示了如何创建和使用自定义匹配器和断言。

尽管基于Assert的机制肯定是过时的且不太面向对象,但它仍然具有它的提倡者。 Hamcrest匹配器提供断言和谓词定义的清晰分隔,而AssertJ断言以紧凑且易于使用的编程样式进行评分。 所以现在您选择太多了……

请注意,这将是本教程有关JUnit测试要点的最后一章。 这并不意味着没有更多要说的了。 恰恰相反! 但这将超出此迷你系列量身定制的范围。 而且您知道他们在说什么: 总是让他们想要更多…

  1. 嗯,我想知道区间边界是否会比下限和范围更直观...

翻译自: https://www.javacodegeeks.com/2014/09/junit-in-a-nutshell-unit-test-assertion.html

junit单元测试断言

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

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

相关文章

keyshot环境素材文件_KeyShot渲染,打光这么打,效果倍儿棒

有过渲染经验的小伙伴们应该知道&#xff0c;除了需要对渲染软件的操作熟悉外&#xff0c;最重要的莫过于“材质”和“灯光”。而这里&#xff0c;就不得不提到KeyShot&#xff0c;作为工业设计行业必须要掌握的技能之一&#xff0c;它在行业当中的地位可谓独树一帜。除了其操作…

redis种类型对应java类型_Redis的五种基本数据类型介绍

Redis作为缓存具有极其丰富的功能&#xff0c;比如计数器、好友关系以及排行榜等等。而Redis之所能够实现如此强大的功能&#xff0c;完全是依赖着它的基本数据结构。今天我们来看看Redis的五种基本的数据结构&#xff0c;分别是字符串(strings)&#xff0c; 散列(hashes)&…

REST /使用提要发布事件

处理事件 当使用多个解耦的服务时&#xff08;例如&#xff0c;在微服务体系结构中 &#xff09;&#xff0c;很有可能需要一种将某种域事件从一个服务发布到一个或多个其他服务的方法。 许多广泛采用的解决方案依赖于单独的基础结构来解决此问题&#xff08;例如事件总线或消…

hibernate自动配置_Hibernate自动冲洗的黑暗面

hibernate自动配置介绍 既然我已经描述了JPA和Hibernate刷新策略的基础知识 &#xff0c;我就可以继续阐明Hibernate的AUTO刷新模式的令人惊讶的行为。 并非所有查询都会触发会话刷新 许多人会认为Hibernate 总是在执行任何查询之前先刷新Session。 虽然这可能是一种更直观的方…

slf4j注解log报错_SpringBoot自定义日志注解,用于数据库记录操作日志,你用过吗?...

大家好&#xff0c;我是程序员7歌&#xff01;今天我将为大家讲解如何通过自定义注解记录接口访问日志。一般的开发中&#xff0c;有两种方式可以记录日志信息&#xff0c;第一种&#xff1a;把接口日志信息保存到日志文件中&#xff0c;第二种&#xff1a;把接口操作日志保存到…

java applet audion_java applet audion

①希罗尤尔和他的飞翼敢达也有着不少拥趸&#xff0c;让我们期待这位美少年在《敢达决战》中的表现吧。②颜值时代&#xff0c;浏览器皮肤也不能输360浏览器耳目一新的设计&#xff0c;高清精美的壁纸&#xff0c;让你上网时更添一份好心情。软件使用1、支持游戏小号2、优化小程…

markdown 流程图_Markdown 进阶技能:用代码画流程图(编程零基础也适用)

这篇文章主要介绍流程图基础以写代码的方式画流程图相比于使用画图工具拖拽画图&#xff0c;用代码画图有什么好处&#xff1f;首先&#xff0c;这种方式非常轻便&#xff0c;无需安装复杂的画图应用。Typora 等多种 Markdown 编辑器自带有画图扩展&#xff08;这也是 Markdown…

java jtree_Java JTree

Java JTree1 Java JTree的介绍JTree类用于显示树结构数据或层次结构数据。JTree是一个复杂的组件。它的最顶部有一个“根节点”&#xff0c;它是树中所有节点的父节点。它继承了JComponent类。2 Java JTree的声明我们来看一下javax.swing.JTree类的声明。public class JTree ex…

akka和rabbitmq_Akka Notes –演员记录和测试

akka和rabbitmq在前两部分&#xff08; 一 &#xff0c; 二 &#xff09;中&#xff0c;我们简要讨论了Actor以及消息传递的工作方式。 在这一部分中&#xff0c;让我们看一下如何修复并记录我们的TeacherActor 。 回顾 这就是我们上一部分中的Actor的样子&#xff1a; class…

完数c++语言程序_C语言经典100题(19)

1上期答案揭晓首先给大家看看上一篇文章C语言经典100题(18)中第三部分编程题的答案&#xff1a;#includeint main(){ int s0,a,n,t; printf("请输入 a 和 n&#xff1a;\n"); scanf("%d%d",&a,&n); ta; while(n>0) { …

古巴:为生产做准备

“它可以在我的本地机器上运行&#xff01;” 如今&#xff0c;这听起来像模因&#xff0c;但仍然存在“开发环境与生产环境”的问题。 作为开发人员&#xff0c;您应始终牢记&#xff0c;您的应用程序有一天将在生产环境中开始运行。 在本文中&#xff0c;我们将讨论一些特定于…

hibernate脏数据_Hibernate脏检查的剖析

hibernate脏数据介绍 持久性上下文使实体状态转换入队 &#xff0c;该实体状态转换在刷新后转换为数据库语句。 对于托管实体&#xff0c;Hibernate可以代表我们自动检测传入的更改并安排SQL UPDATE。 这种机制称为自动脏检查 。 默认的脏检查策略 默认情况下&#xff0c;Hibe…

php组成,php接口有几部分组成?

程序接口&#xff0c;由一套陈述、功能、选项、其它表达程序结构的形式、以及程序师使用的程序或者程序语言提供的数据组成PHP接口(interface)的特点1、接口的方法必须是公开的。2、接口的方法默认是抽象的&#xff0c;所以不在方法名前面加abstract。3、接口可以定义常量&…

java 解析日期格式_日期/时间格式/解析,Java 8样式

java 解析日期格式自Java 几乎 开始以来&#xff0c;Java开发人员就通过java.util.Date类&#xff08;自JDK 1.0起&#xff09;和java.util.Calendar类&#xff08;自JDK 1.1起 &#xff09;来处理日期和时间。 在这段时间内&#xff0c;成千上万&#xff08;甚至数百万&#x…

php第三方登录代码,thinkPHP5项目中实现QQ第三方登录功能

本文实例讲述了thinkPHP5项目中实现QQ第三方登录功能。分享给大家供大家参考&#xff0c;具体如下&#xff1a;最近用thinkPHP 5框架做了一个婚纱店的项目&#xff0c;在开发过程中需要用到第三方登录&#xff0c;腾讯官方给的案例是几个文件相互包含实现的&#xff0c;放到tp5…

mac 显示隐藏文件_如何在Mac上显示隐藏文件?苹果mac显示隐藏文件夹方法

与任何操作系统一样&#xff0c;macOS会将重要文件隐藏起来&#xff0c;以防止意外删除它们并因此而损坏系统。但是&#xff0c;在某些情况下&#xff0c;您可能需要在Mac上显示隐藏文件&#xff0c;例如&#xff0c;浏览“ 库”文件夹并清除旧日志&#xff0c;缓存或其他垃圾文…

分布式虚拟跟踪

跟踪提供了对系统的可见性&#xff0c;使开发人员和操作人员可以在运行时观察应用程序。 当系统不断增长并与更多微服务进行交互时&#xff0c;跟踪变得非常有价值。 在这样的环境中&#xff0c;这些痕迹非常棒&#xff0c;可以定位导致性能下降的故障和瓶颈。 在这篇文章中&a…

php 删除数组的空元素,php删除数组空元素的方法_后端开发

php如何实现自动跳转_后端开发php实现自动跳转的方法&#xff1a;1、通过php内置函数“header”&#xff0c;将http响应头中的“Location”设置为要跳转的URL即可&#xff1b;2、可以在javascript代码中将“window.location.href”指向要跳转的URL即可。php删除数组空元素的方法…

map for循环_JavaScript 用 for 循环太 low?你是不是有什么误解

天要吐槽下&#xff0c;我时不时地看到有些文章说“循环语句不好&#xff0c;你应该用 filter&#xff0c;map 和 reduce ”——每次看到有文章鼓吹&#xff0c;所有需要循环的场景一律用这几个函数式方法&#xff0c;我都恨得牙痒痒。没错&#xff0c;这些函数式方法确实有它们…

简单工程验收单表格_中铁超大型工程项目-123个精细化管理手册配套表格附件,超全...

中铁超大型工程项目-123个精细化管理手册配套表格附件&#xff0c;超全&#xff01;什么是项目精细化&#xff1f;答&#xff1a;工程项目精细化管理是一个系统的管理体系&#xff0c;包含一系列管理制度和办法&#xff0c;除了《工程项目精细化管理办法》这个纲领性文件外&…