在测试中构造对象通常是一件艰苦的工作,通常会产生大量可重复且难以阅读的代码。 有两种用于处理复杂测试数据的常见解决方案: 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

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

  1. 数据库 测试数据生成_测试数据生成器和对象母亲:另一种外观

    数据库 测试数据生成 在测试中构造对象通常是一项艰巨的工作,通常会产生大量可重复且难以阅读的代码. 有两种用于处理复杂测试数据的常见解决方案: Object Mother和Test Data Buil ...

  2. JAVA构造对象的几种方式(构建器、构造器)

    大家好,我是烤鸭: 今天说一下初始化对象的几种方式:   1.    多参数构造器 2.    构建器  3.    构造器后 + get/set方法 举个例子: 这里有个机构entity,提供一个默 ...

  3. C++编程思想 第2卷 第10章 设计模式 构建器模式:创建复杂对象

    构建器 Builder 模式的模板是将对象的创建与它的 表示法 分开 意味着 创建保持原状  但是产生对象 的表示法 不同 作为模型的自行车按照其内心来选择零部件组装一辆自行车 一个构建器与每个自行车 ...

  4. 11. 对象创建模式之 Builder模式(构建器)(不常用)

    1. 动机 在软件系统中,有时候面临着"一个"复杂对象"的创建工作,其通常由各个部分的子对象用一定的算法构成:由于需求的变化,这个复杂对象的各个部分经常面临着剧烈的变化, ...

  5. 11Builder(构建器)模式

    技术交流QQ群:1027579432,欢迎你的加入! 1.Builder(构建器)模式动机 在软件系统中,有时候面临着一个复杂对象的创建工作,其通常由各个部分的子对象用一定的算法构成.由于需求的变化, ...

  6. java 管道设计_使用管道流实现Java 8阶段构建器

    Step builder多阶段步骤构造器模式是一种对象创建软件设计模式.与传统构建器模式进行比较时,步骤构建器模式提供了一些简洁的好处.Step Builder模式的主要优势之一是为客户提供有关如何使 ...

  7. java查询类提供的方法_查询一个类的方法和构建器的JAVA程序

    查询一个类的方法和构建器的JAVA程序 JAVA语言的类库中的类不计其数,其中设计的方法更是不胜其多,这是任何一个教科书--无论它是多么详尽--所不能穷尽的.如果我们要使用某个类,要查询它的构建器的用 ...

  8. Microsoft宣布为Power BI提供AI模型构建器,关键驱动程序分析和Azure机器学习集成...

    微软的Power BI现在是一种正在大量结合人工智能(AI)的商业分析服务,它使用户无需编码经验或深厚的技术专长就能够创建报告,仪表板等.近日西雅图公司宣布推出几款新的AI功能,包括图像识别和文本分析 ...

  9. 构建器设计模式_创新设计模式:构建器模式

    构建器设计模式 以前我们看过工厂和抽象工厂模式. 这些模式可以达到目的,并且确实有用,但是在某些用例中,我们必须创建一个非常复杂的对象,并且创建它需要不同的步骤,每个步骤都需要不同的操作. 在这种情况 ...

最新文章

  1. 把canvas放在盒子内_如何将您的专业知识放在盒子中并出售
  2. 2021年6月程序员平均工资 15052,你给行业拖后腿了吗?
  3. URL编码将“&”(&符号)视为“&”HTML实体
  4. 用pip安装tensorflow报错SyntaxError: invalid syntax
  5. Redis 未授权访问漏洞
  6. leetcode739 每日温度
  7. [Unity] ACT 战斗系统学习 7:使用 ScriptableObject 制作角色属性 2
  8. 【Java数据结构与算法】第八章 快速排序、归并排序和基数排序
  9. Introduction to Computer Networking学习笔记(二十):delay guarantees延迟速度保证
  10. Axure rp 9安装包下载地址、汉化文件、授权码(包括下载地址)
  11. 【自动化测试工具】QTP/UFT入门
  12. mysql mgr 启动_使用MySQL Shell创建MGR
  13. hapi.js_Hapi.js入门
  14. 接入科大讯飞语音听写,增加语音动画,类似京东语音搜索功能
  15. 把时间沉淀到自己的热爱里 | Kagol 的 2022 年终总结
  16. 国内外抠图平台性能调研
  17. 输入一行英文句子,统计其中的单词个数。例如,输入”How are you.”,则输出3。
  18. 虚拟服务器怎样做网站,怎么用虚拟主机做网站
  19. MySQL 集群(三):MySQL + Mycat 实现读写分离,主备切换集群
  20. 云计算专业怎么样,大学应届生学的话难不难?

热门文章

  1. 2018蓝桥杯省赛---java---C---2(猴子分香蕉)
  2. Mybatis中使用Dao实现类实现增删改查【实际开发中使用代理dao】
  3. linux u32,如何在程序中使用u32这个类型啊。
  4. centos7离线安装oracle11g,CentOS 7.5离线安装Oracle 11gR2
  5. ubuntu 13.04 mysql_Ubuntu13.04 下MySQL5.6安装过程
  6. php accesscontrolalloworigin,设置Access-Control-Allow-Origin实现跨域访问
  7. 小小涉及OpenFeign原理:Could not extract response: no suitable HttpMessageConverter found for response type
  8. spring boot自测_将测微仪与Spring Boot 2一起使用
  9. java 性能调优_Java性能调优调查结果(第三部分)
  10. if else 工厂模式_没有IF-ELSE的工厂