测试数据构建器和对象母亲:另一种眼神

在测试中构造对象通常是一件艰苦的工作,通常会产生大量可重复且难以阅读的代码。 有两种用于处理复杂测试数据的常见解决方案: Object MotherTest Data Builder 。 两者都有优点和缺点,但是(巧妙地)结合可以为您的测试带来新的质量。

注意:关于Object MotherTest Data Builder ,已经有很多文章可以找到,因此我将使我的描述非常简洁。

对象母亲

不久,“ 对象母亲”是一组工厂方法,允许我们在测试中创建类似的对象:

// Object Mother
public class TestUsers {public static User aRegularUser() {return new User("John Smith", "jsmith", "42xcc", "ROLE_USER");}// other factory methods}// arrange
User user = TestUsers.aRegularUser();
User adminUser = TestUsers.anAdmin();

每次需要数据变化稍有不同的用户时,都会创建新的工厂方法,这会使“ Object Mother随时间增长。 这是Object Mother的缺点之一。 可以通过引入Test Data Builder解决此问题。

测试数据生成器

Test Data Builder使用Builder模式在单元测试中创建对象。 一的简短提醒Builder

构建器模式是对象创建软件设计模式。 […]构建器模式的目的是找到可伸缩构造函数反模式的解决方案。

让我们看一下Test Data Builder的示例:

public class UserBuilder {public static final String DEFAULT_NAME = "John Smith";public static final String DEFAULT_ROLE = "ROLE_USER";public static final String DEFAULT_PASSWORD = "42";private String username;private String password = DEFAULT_PASSWORD;private String role = DEFAULT_ROLE;private String name = DEFAULT_NAME;private UserBuilder() {}public static UserBuilder aUser() {return new UserBuilder();}public UserBuilder withName(String name) {this.name = name;return this;}public UserBuilder withUsername(String username) {this.username = username;return this;}public UserBuilder withPassword(String password) {this.password = password;return this;}public UserBuilder withNoPassword() {this.password = null;return this;}public UserBuilder inUserRole() {this.role = "ROLE_USER";return this;}public UserBuilder inAdminRole() {this.role = "ROLE_ADMIN";return this;}public UserBuilder inRole(String role) {this.role = role;return this;}public UserBuilder but() {return UserBuilder.aUser().inRole(role).withName(name).withPassword(password).withUsername(username);}public User build() {return new User(name, username, password, role);}
}

在我们的测试中,我们可以如下使用构建器:

UserBuilder userBuilder = UserBuilder.aUser().withName("John Smith").withUsername("jsmith");User user = userBuilder.build();
User admin = userBuilder.but().withNoPassword().inAdminRole();

上面的代码看起来非常不错。 我们有一个流畅的API,可以提高测试代码的可读性,并且可以肯定地消除了使用Object Mother需要多种工厂方法来处理测试中需要的对象变化的问题。

请注意,我添加了一些默认属性值,这些默认值可能与大多数测试无关。 但是,由于我们将它们定义为公共常量,因此可以在断言中使用它们。

注意:本文使用的示例相对简单。 它用于可视化解决方案。

对象母亲和测试数据生成器结合

两种解决方案都不完美。 但是,如果我们将它们结合起来呢? 想象一下, Object Mother返回一个Test Data Builder 。 有了这个,您就可以在调用终端操作之前操纵构建器状态。 这是一种模板。

看下面的例子:

public final class TestUsers {public static UserBuilder aDefaultUser() {return UserBuilder.aUser().inUserRole().withName("John Smith").withUsername("jsmith");}public static UserBuilder aUserWithNoPassword() {return UserBuilder.aUser().inUserRole().withName("John Smith").withUsername("jsmith").withNoPassword();}public static UserBuilder anAdmin() {return UserBuilder.aUser().inAdminRole().withName("Chris Choke").withUsername("cchoke").withPassword("66abc");}
}

现在, TestUsers提供了用于在我们的测试中创建类似测试数据的工厂方法。 它返回一个构建器实例,因此我们能够根据需要在测试中快速而完美地修改对象:

UserBuilder user = TestUsers.aUser();
User admin = user.but().withNoPassword().build();

好处是巨大的。 我们有一个用于创建相似对象的模板,如果我们需要在使用返回对象之前修改返回对象的状态,则我们拥有构建器的强大功能。

丰富测试数据生成器

考虑以上内容时,我不确定是否真的需要单独的Object Mother 。 我们可以轻松地将方法从Object Mother直接移动到Test Data Builder

public class UserBuilder {public static final String DEFAULT_NAME = "John Smith";public static final String DEFAULT_ROLE = "ROLE_USER";public static final String DEFAULT_PASSWORD = "42";// field declarations omitted for readabilityprivate UserBuilder() {}public static UserBuilder aUser() {return new UserBuilder();}public static UserBuilder aDefaultUser() {return UserBuilder.aUser().withUsername("jsmith");}public static UserBuilder aUserWithNoPassword() {return UserBuilder.aDefaultUser().withNoPassword();}public static UserBuilder anAdmin() {return UserBuilder.aUser().inAdminRole();}// remaining methods omitted for readability}

因此,我们可以在单个类中维护User数据的创建。

请注意,在此Test Data Builder是一个测试代码。 如果我们在生产代码中已经有一个生成器,那么创建一个Object Mother返回一个Builder实例听起来是一个更好的解决方案。

可变对象呢?

当涉及可变对象时, Test Data Builder方法可能存在一些缺点。 在许多应用程序中,我主要处理可变对象(又称为beansanemic data model ),也许你们中的许多人也这样做。

从理论上讲, Builder模式用于创建不变的价值对象。 通常,如果我们处理可变对象,乍一看, Test Data Builder可能看起来像是重复的:

// Mutable class with setters and getters
class User {private String name;public String getName() { ... }public String setName(String name) { ... }// ...
}public class UserBuilder {private User user = new User();public UserBuilder withName(String name) {user.setName(name);return this;}// other methodspublic User build() {return user;}
}

在测试中,我们可以创建一个如下用户:

User aUser = UserBuiler.aUser().withName("John").withPassword("42abc").build();

代替:

User aUser = new User();
aUser.setName("John");
aUser.setPassword("42abc");

在这种情况下,创建Test Data Builder是一个折衷方案 。 它需要编写更多需要维护的代码。 另一方面,可读性大大提高。

摘要

在单元测试中管理测试数据并非易事。 如果找不到合适的解决方案,那么最终将获得大量难以理解和维护的样板代码。 另一方面,没有解决该问题的灵丹妙药。 我尝试了许多方法。 根据问题的大小,我需要选择一种不同的方法,有时在一个项目中结合使用多种方法。

您如何处理测试中的数据构建?

资源资源

  • Petri Kainulainen: 编写干净的测试–被认为有害的新方法
  • Growing Object-Oriented Software, Guided by Tests –第22章: Constructing Complex Test Data

翻译自: https://www.javacodegeeks.com/2014/06/test-data-builders-and-object-mother-another-look.html

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

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

相关文章

java 递归函数_浅谈java递归函数

递归函数的定义:递归函数即自调用函数,在函数体内直接或间接的调用自己,即函数的嵌套是函数本身。打个生动的比喻:我们可以把” 递归 “比喻成 “查字典 “,当你查一个词,发现这个词的解释中某个词仍然不懂…

这几周的一些破事

1:明基几百人同时大跳槽到另外一家公司明基上海有限公司(苏州分公司),当然也包括我了.2:BQM GSR开始了,搞PES,搞ProtoTYpe,搞PIS,没有啥意思,对工作没有什么热情了,精力也不是很集中.3:日本的破CR迟迟不肯上线,时不时来骚扰一下,烦的要死,为什么我们就该这么忙,可恨.4:看到哪些…

oracle data guard --理论知识回顾02

继上一篇 管理影响物理standby的事件 1 创建表空间或数据文件初始化参数standby_file_management用来控制是否自动将primary数据库增加表空间或数据文件的改动,传播到standby服务器AUTO/MANUALAUTO:自动传播到standbyMANUAL:需要手动复制文件到standby,并…

Java 8 LongAdders:管理并发计数器的正确方法

我只是喜欢新玩具,而Java 8有很多 。 这次我想谈谈我的最爱之一-并发加法器。 这是一组用于管理由多个线程编写和读取的计数器的新类。 新的API有望显着提高性能,同时仍然使事情变得简单明了。 自从多核架构问世以来人们一直在管理并发计数器&#xff0…

asp: menu 父级选中

最近比较忙,其中一个项目中用到了菜单,结果User比较难搞定,非要什么有父级菜单选中的样式,就像windows的菜单效果差不多的那种,鼠标飘在子结点上,父结点也要保留反白的选中样式,唉,U…

java 内部类 单例_Java单例模式的几种实现

一:静态内部类实现单例模式原理:通过一个静态内部类定义一个静态变量来持有当前类实例,在类加载时就创建好,在使用时获取。缺点:无法做到延迟创建对象,在类加载时进行创建会导致初始化时间变长。public cla…

JS中ptototype和__proto__的关系

学到原型的时候感觉头都大了/(ㄒoㄒ)/~~ 尤其是ptototype和__proto__ 傻傻分不清 通过多番查找资料,根据自己的理解,总结如下: 一、构造函数: 构造函数:通过new关键字可以用来创建特定类型的对象的函数。比如像Obje…

函数与匿名函数

函数与匿名函数一、函数 在计算机科学中,函数是程序负责完成某项特定任务的代码单元,具有一定的独立性。 1.函数的定义 在Python中,函数在定义时需要满足这样的规则: 使用关键字def引导;def后面是函数的名称&#xff0…

[最短路]飞行

题目描述 WFYZ的校园很大,这里生活着很多生物,比如住在钟楼上的的鸽子,其中鸽子冉冉和她的妹妹凝凝白天在不同的地方吃虫,而在晚上她们都回到钟楼休息。她俩是两只懒鸟,于是提出了一个计划,尽量减少她们在飞…

Java状态和策略设计模式之间的差异

为了在Core Java应用程序中正确使用状态和策略设计模式,对于Java开发人员清楚地了解它们之间的区别很重要。 尽管状态和策略设计模式的结构相似,并且都基于开放式封闭设计原则,从SOLID设计原则表示为“ O”,但它们在意图上完全不同…

【原创】系统分析师--任重而道远

今天查到了系统分析师的成绩。有坏消息,也有好消息。 坏消息当然就是意料之中的没过了,好消息是我基本上没有花什么时间去准备,但是结果却比想象中的要好。 案例分析这一场过了。基础知识差几分,论文只是及格分的一半&#xf…

XML文档的简易增删查改

dom4j解析一、利用dom4j操作元素节点 1.查询第一本书的书名,并输出到控制台 2.给第一本书添加一个特价节点, 并修改第一本书的售价节点的内容为19.8元 3.删除第二本书的作者节点二、利用dom4j操作属性节点 1.给第一本书添加一个属性,如:出版社…

廖雪峰Java3异常处理-1错误处理-2捕获异常

1捕获异常 1.1 finally语句保证有无错误都会执行 try{...}catch (){...}finally{...} 使用try...catch捕获异常可能发生异常的语句放在try{...}中使用catch捕获对应的Exception及其子类1.2 捕获多个异常 try{...} catch() {...} catch(){...}finally{..} 使用多个catch子句&…

更新数据库

方法一:在对SQL数据库进行更新时,用CommandBuilder对像来自动构建sql命令,来起到更新的作用;这种方法用起来比较方便,具体代码如下: 以下代码都在xp系统下测试通过 环境:vs.net2005 \ sql server 2000\xpus…

在崩溃或断电后测试Lucene的索引耐久性

Lucene有用的事务功能之一是索引持久性 ,它可以确保一旦成功调用IndexWriter.commit ,即使操作系统或JVM崩溃或断电,或者您杀死-KILL JVM进程,重启后索引也将保持完整(未损坏),并将反映崩溃前的…

Bash 脚本 set 命令教程

http://www.ruanyifeng.com/blog/2017/11/bash-set.html set命令是 Bash 脚本的重要环节,却常常被忽视,导致脚本的安全性和可维护性出问题。本文介绍它的基本用法,让你可以更安心地使用 Bash 脚本。 一、简介 我们知道,Bash 执行脚…

Jmeter、postman、python 三大主流技术如何操作数据库?

1、前言 只要是做测试工作的,必然会接触到数据库,数据库在工作中的主要应用场景包括但不限于以下: 功能测试中,涉及数据展示功能,需查库校验数据正确及完整性;例如商品搜索功能 自动化测试或性能测试中&a…

[转][HTML]css属性

1、字体属性(type)font-family(使用什么字体)font-style(字体的样式,是否斜体):normal/italic/obliquefont-variant(字体大小写):normal/small-capsfont-weight(字体的粗细):normal/bold/bolder/lithterfont-size(字体的大小)&…

7-OKHttp使用详解,步骤挺详细的,适合初学者使用!

OKHttp使用详解,步骤挺详细的,适合初学者使用! 一,OKHttp介绍 okhttp是一个第三方类库,用于android中请求网络。 这是一个开源项目,是安卓端最火热的轻量级框架,由移动支付Square公司贡献(该公司还贡献了Picasso和Leak…

变量与递归函数

变量与递归函数一、变量 变量分为全局变量和局部变量。 python中全局变量和局部变量的最大区别在于局部变量只能通过函数去访问,而全局变量可以直接访问。 举个例子: name linxu #全局变量 def change_name():namepython #局部变量print(name) …