数据库 测试数据生成

在测试中构造对象通常是一项艰巨的工作,通常会产生大量可重复且难以阅读的代码。 有两种用于处理复杂测试数据的常见解决方案: 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. 数据库 测试数据生成_我们的测试数据生成器如何使假数据看起来真实

    数据库 测试数据生成 by Tom Winter 汤姆·温特(Tom Winter) 我们的测试数据生成器如何使假数据看起来真实 (How our test data generator makes ...

  2. mysql序列号生成_一文看懂mycat的6种全局序列号实现方式

    概述 在实现分库分表的情况下,数据库自增主键已无法保证自增主键的全局唯一.为此,MyCat 提供了全局sequence,并且提供了包含本地配置和数据库配置等多种实现方式.下面对这几种实现方式做一下介绍 ...

  3. 遍历对象属性_细说JS遍历对象属性的N种方法

    本人详细介绍了JS遍历对象N种方法,欢迎关注收藏. 遍历对象属性有五种方法,下图为一个场景对比图. 可以注意到两点: 只有for ... in 才可以遍历原型链属性,且只能遍历可枚举属性. Objec ...

  4. 测试数据生成工具datafaker使用

    1.工具使用场景 在软件开发测试过程,经常需要测试数据.这些场景包括: 1.1 后端开发 新建表后,需要构造数据库测试数据,生成接口数据提供给前端使用. 1.2数据库性能测试 生成大量测试数据,测试数 ...

  5. datafactory生成mysql数据_测试数据生成工具DataFactory的使用

    DATA FACTORY的使用 Data Factor y是一个数据库测试数据生成工具. Data Factory主要可以利用在以下两个方面:1.按照数据表中要求数据的格式,快速产生标准或不标准的测试 ...

  6. 在线MySQL,SQL Server建表语句生成JSON测试数据工具

    在线MySQL,SQL Server建表语句生成JSON测试数据工具 在线MySQL,SQL Server建表语句生成JSON测试数据工具 本工具可以将SQL建表语句生成JSON测试数据,支持MySQ ...

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

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

  8. db4o_8.0对象数据库官方文档翻译_学习笔记三

    紧接上篇:db4o_8.0对象数据库官方文档翻译_学习笔记二 3. Object Manager Enterprise Overview(OME视图)即OME插件的使用 If you did not  ...

  9. MySQL如何生成大批量测试数据

    前言 不管是为了做生产数据脱敏,还是为了验证某个技术是否符合期望,亦或是做简单的集成测试,短时间内生成大批量测试数据是很有必要的,但就算是测试数据也要假的像样点嘛,数据不够假,你放到演示环境,怎么给客 ...

最新文章

  1. 3D视觉工坊中秋国庆贺礼!
  2. Ubuntu安装python3.7,并将python链接指向python3.7
  3. java dozer 深度_java – Dozer深度映射设置为Set
  4. windows7使用vhd虚拟磁盘
  5. VTK:IO之ReadDICOMSeries
  6. 与Netflix合作 美电视运营商推出4K频道
  7. 2016-7-4收藏夹接口
  8. HTML+CSS+JS实现 ❤️圣诞抓礼物小游戏❤️
  9. mysql事务日志备份_事务日志备份 (SQL Server)
  10. 能改变原生web前端元素样式的water.css
  11. MASQUERADE
  12. java jquery用的多吗_[Java教程]如果不用jQuery,Ajax你还能写出多少?
  13. MarkDown说明
  14. kepware OPC UA通讯问题
  15. jquery和ajax实战教程电子书,《jQuery和Ajax实战教程》邵山欢_孔网
  16. 计算机文化基础简答题
  17. r7 2700X装Linux,R7-2700X配什么主板?AMD锐龙7 2700X主板推荐 (全文)
  18. Cocos2d-x学习笔记(五)仿真树叶飘落效果的实现(精灵旋转、翻转、钟摆运动等综合运用)
  19. 2018 Google 开发者大会.md
  20. Proxmox VE 7.2 使用qemu-img转换磁盘格式

热门文章

  1. ssl2331OJ1373-鱼塘钓鱼 之1【纯贪心】
  2. 【DP】Sleeping Cows P(P7154)
  3. 【dfs】【hash】有趣的英语角(2015特长生 T2/luogu 1019)
  4. codeforces E. Picking Strings 构造
  5. 动态规划训练23 [Making the Grade POJ - 3666 ]
  6. 4、mybatis通过配置类Configuration 实现初始化
  7. Java 证书pem转KeyStore、jks文件
  8. MongoDb连接表的查询
  9. 阿里巴巴对Java编程【OOP规约】的规约
  10. 接口限流算法:漏桶算法amp;令牌桶算法