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

news/2024/7/10 3:21:48 标签: java, python, 设计模式, vue, javascript
软件测试

软件测试

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

注意:关于Object MotherTest Data Builder ,已经有很多文章可以找到,因此我将保持简洁。

对象母亲

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

java">// 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的示例:

java">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);
    }
}

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

java">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 。 有了这个,您就可以在调用终端操作之前操纵构建器状态。 这是一种模板。

看下面的例子:

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

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

好处是巨大的。 我们有一个用于创建类似对象的模板,如果我们需要在使用返回的对象之前修改其状态,则可以使用生成器。

丰富测试数据生成器

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

java">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 readability

    private 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可能看起来像是重复的:

java">// 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 methods

    public User build() {
        return user;
    }
}

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

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

代替:

java">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.niftyadmin.cn/n/1707927.html

相关文章

hibernate_测试Hibernate的最低配置

hibernate介绍 在我以前的文章中,我宣布了我打算创建个人Hibernate课程的意图。 首先要做的是最小的测试配置。 这些示例与Hibernate 4有关。 您只需要Hibernate 在实际的生产环境中,您不会单独使用Hibernate,因为您可以将其集成到JEE或Sprin…

android手机避免不必要的流量,如何防止手机偷跑流量?小米手机防止流量偷跑的设置技巧...

近日,央视报道许多手机后台偷跑流量,其中包括小米手机,很多人在使用手机时经常也会遇到流量偷跑的情况,其实对于小米手机来说,这样情况是可以避免的。那么那么如何防止手机偷跑流量?其实对于小米用户来讲,…

数据结构栈的知识点_数据知识栈

数据结构栈的知识点并发不适合胆小者 我们都知道并发编程很难正确实现。 这就是为什么在执行线程任务之后要进行大量的设计和代码审查会议。 您永远不会将并发问题分配给经验不足的开发人员。 仔细分析问题空间,提出设计,并记录和审查解决方案。 这就是通…

jboss1.7_检查Red Hat JBoss BRMS部署架构的规则和事件(第二部分)

jboss1.7(文章来宾与北美红帽公司高级中间件顾问约翰赫洛克( John Hurlocker)合着) 在这周的技巧中,我们将放慢速度,并仔细研究可能的Red Hat JBoss BRMS部署体系结构。 当我们谈论部署架构时,我们指的是在企业中部署规则和/或事件项目所必需…

html li显示不出列表效果,html – 每个li的不同列表样式图像

我有一些ul和li的问题……我想用3 li创建一个无序列表,每个li都有一个不同的列表样式图像…我写了这段代码,但图片没有出现……你能帮助我吗?谢谢!编辑:发布更新HTML代码;)Some text here...Some text here...Some text here...#right_main u…

jersey_教程–带有Jersey和Spring的Java REST API设计和实现

jersey想要在Java中使用REST? 然后您来对地方了,因为在博客文章中,我将向您介绍如何“美丽”地设计REST API,以及如何使用Jersey框架在Java中实现它。 在本教程中开发的RESTful API将为存储在MySql数据库中的播客资源演示完整的Cr…

python将html页面保存为字符串,python – 将html字符串插入到BeautifulSoup对象中

最简单的方法,如果你已经有一个html字符串,就是插入另一个BeautifulSoup对象.from bs4 import BeautifulSoupdoc test1soup BeautifulSoup(doc, html.parser)soup.div.append(BeautifulSoup(insert1, html.parser))print soup.prettify()输出:test1insert1更新1这…

泛型 java_使用Java泛型的模板方法模式示例

泛型 java如果您发现除了某些部分外,您的许多例程完全相同,那么您可能需要考虑使用Template Method来消除容易出错的代码重复。 这是一个示例:下面是两个做类似事情的类: 实例化并初始化Reader以从CSV文件读取。 阅读每一行并将其…