构建器模式:适用于代码,适用于测试

我发现生成器设计模式偶尔在代码中有用,但在测试中经常有用。 本文简要概述了该模式,然后介绍了在测试中使用该模式的一个有效示例。 请参阅github中的代码。

生成器模式的背景

根据GoF的书 ,构建器设计模式用于“将复杂对象的构造与其表示分离,以便同一构造过程可以创建不同的表示”。 像大多数GoF书中一样,这是一个准确而乏味的描述。

乔什·布洛赫(Josh Bloch)在他的《 有效的Java》一书中,为建造者提出了一种更有趣的用法。 他的方法试图解决的问题是,当一个类具有“不止几个”参数时,这些参数通常是通过构造函数设置的,其中许多参数可能是可选的。 典型的解决方案是

  • 伸缩构造函数模式,在该模式中,您将为构造函数仅提供必需的参数,并为其他构造函数提供可选参数的变体,最终形成具有所有可选参数的构造函数。
    这可以工作,但会导致一个相当混乱的解决方案,该解决方案可能包含大量构造函数以覆盖所有排列
  • 一个简单的构造函数(例如,仅用于必需的参数),由setter方法支持可选参数(JavaBeans方法)。 但是,这可能会使对象在构造过程中处于不一致状态,并且由于无法将字段定为final ,因此当然会阻止不变性。
  • 使用一个生成器。 这是Bloch建议的方法。 客户端创建一个生成器(通常使用无参数的构造器),然后在最终调用一个build方法之前,调用类似setter的方法来获取感兴趣的值(其余的假定为默认值)。

几年前,我参加了一次演讲 ,其中Ted Young讨论了通过将构建器模式用于测试对象的构建来使构建器模式更进一步,下面讨论的是这种方法。 [更新:请在此处查看Ted对这篇文章的回复]

使用Builder模式构造测试装置

使用Builder可以更轻松,更清晰地创建测试装置。

我通常使用此Builder方法进行测试的对象类型是域模型对象,例如Account,User,Widget或其他对象。 我支持使此类对象不变 。
例如:

public final class Account {private final Integer id;private final String name;private final AccountType type;private final BigDecimal balance;private final DateTime openDate;private final Status status;public Account(Integer id, String name, AccountType type,BigDecimal balance, DateTime openDate, Status status) {this.id = id;this.name = name;this.type = type;this.balance = balance;this.openDate = openDate;this.status = status;}public Integer getId() {return id;}    //other getters, toString(), equals() and hashCode() omitted for brevity//no setters
}

使用此类,您经常会遇到Bloch讨论的问题。 在此示例中,我们有一个强制您设置所有值的构造函数,但是我们也可以有很多变体,其中一些值可以省略以使用默认值。 因此,为测试创建此类的实例可能会有些痛苦,如果它具有比此简单示例更多的字段,则更加痛苦。 您甚至不得不为可能不需要测试的字段提供值。 这也使得很难知道哪些值实际上是测试所需要的,哪些值纯粹是为了编译。

建设者可以提供帮助。

public class AccountBuilder {//account fields with default valuesInteger id = 1;String name = "default account name";AccountType type = AccountType.CHECKING;BigDecimal balance = new BigDecimal(0);DateTime openDate = new DateTime(2013, 01, 01, 0, 0, 0);Status status = Status.ACTIVE;public AccountBuilder() {}public AccountBuilder withId(Integer id) {this.id = id;return this;}public AccountBuilder withName(String name) {this.name = name;return this;}public AccountBuilder withType(AccountType type) {this.type = type;return this;}public AccountBuilder withBalance(BigDecimal balance) {this.balance = balance;return this;}public AccountBuilder withOpenDate(DateTime openDate) {this.openDate = openDate;return this;}public AccountBuilder withStatus(Status status) {this.status = status;return this;}public Account build() {return new Account(id, name, type, balance, openDate, status);}
}

现在,您可以创建一个Account对象以更轻松地进行测试。

关于使用Builder进行测试的注意事项

    • 默认值

构建器中使用的缺省值是避免出现异常的便利。 如果您的测试需要特定的测试值,则最好明确设置它们,而不要依赖任何默认值。 它使您的测试意图更加清晰,并且在您需要更改默认值(例如由于业务需求变化)而使无意中止测试的风险最小化的情况。

    • 非最终领域

域模型类本身是不可变的,因此具有最终字段。 根据设计,生成器中的所有字段都是非最终的。 因此,构建器不是线程安全的。 因此,请勿重复使用Builders; 而是为每个测试创建一个新实例。

    • 方法顺序不大

在大多数情况下,在构建器上调用方法的顺序应该不重要,并且在调用build()之前不会构造该对象。 这使构建器更易于使用,并避免了意外的意外。
此经验法则有明显且可以接受的例外。
例如打电话

Account account = new AccountBuilder().withType(AccountType.SAVING).withType(AccountType.CHECKING).build();

很傻,但被允许。 它只会为您提供类型检查的权限。
很好,但是请尽量避免引起混乱的细微原因,例如,如果您的集合中可以添加某些内容,或者替换了整个集合(因此请删除以前的添加内容)。

使用Builder进行测试的优势

    • 易于阅读

以下声明不是特别清楚:

Account account = new Account(1, "test", 10, ...);

该声明更加清晰:

Account account = new AccountBuilder().withId(1).withName("test").withBalance(10).build();

正如Bloch所说,“ Builder模式模拟命名的可选参数”。

    • 仅指定与您的测试实际相关的值

如果您的测试仅涉及帐户余额和状态:

Account account = new AccountBuilder().withBalance(new BigDecimal(-100)).withStatus(Status.OVERDRAWN).build();

与必须在Accounts构造函数中指定每个值相反。

    • 创建无效对象的能力

域模型类的构造函数可能会(希望!)强迫您创建有效的对象。 在测试中,您可能要故意创建无效的对象以进行测试。

进一步的增强

便利方法

您可以为测试中使用的常见方案添加便捷方法。
例如

public AccountBuilder withNegativeBalance() {this.balance = new BigDecimal(-100);return this;}

灯具类

除了使用Builder类之外,我还发现拥有一个关联的Fixtures类很有用,该类提供用于测试的预构建实例。 这些可以利用Builder对象进行构造(尽管也没有什么可以阻止您使用原始构造函数)。
例如

public class AccountFixtures {//a shortcut to creating a basic Account objectpublic final Account ACCOUNT = new AccountBuilder().build();public final Account OVERDRAWN_CHECKING_ACCOUNT = new AccountBuilder().withType(AccountType.CHECKING).withNegativeBalance().build();public final Account CLOSED_SAVING_ACCOUNT = new AccountBuilder().withType(AccountType.SAVING).withZeroBalance().withStatus(Status.CLOSED).build();
}

参考: 构建器模式:适用于代码,非常适合我们的JCG合作伙伴 Shaun Abram在Shaun Abram博客博客中进行的测试。

翻译自: https://www.javacodegeeks.com/2013/06/builder-pattern-good-for-code-great-for-tests.html

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

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

相关文章

如何判断html页面停止滚动?

写在开始的话 查遍的文献,没有找到js或者jquery定义好的方法可用,最后迫不得已自己写了个方法。(如果哪位同学知道有其他方法,欢迎讨论) 代码 var count_index 0;$(window).scroll(function(e) {if(count_index 0) {…

HTML5教程之-文件拖拽功能实现

现在打开优酷的网站我们再观看视频时如果拖动滚动条,页面会自动出现一个小的播放窗口,而这个播放窗口是可以说随便拖拽的。这种对图片或者链接的拖拽效果是怎么实现的?其实用HTML5就很容易实现了,因为HTML5中有个html5中默认对图片…

php 打乱数组顺序_PHP实现大转盘抽奖算法

php中文网最新课程每日17点准时技术干货分享本文通过具体的实例向大家介绍了PHP语言实现大转盘抽奖算法,希望对大家学习PHP抽奖有所帮助。流程:1.拼装奖项数组;2.计算概率;3.返回中奖情况。代码如下:中奖概率 v 可以…

linux线程基础篇----线程同步与互斥

linux线程基础----线程同步与互斥 一、同步的概念 1.同步概念 所谓同步,即同时起步,协调一致。不同的对象,对“同步”的理解方式略有不同。如,设备同步,是指在两个设备 之间规定一个共同的时间参考;数据库同…

分布式系统开发注意事项

开发分布式软件系统时,要考虑许多因素。 如果您甚至不知道第一句话中我在说什么,那么让我为您提供一些见解,示例以及有关分布式系统的实例。 总览 分布式系统是指多个物理硬件设备与单独的离散用户交互并通过这些硬件设备协作以为这些离散的…

只需5步,轻松创建HTML5离线应用

1 – 添加 HTML5 doctype第一件要做的事情是创建一个符合规范的 HTML5 文档。HTML5 doctype 相比于 xhtml 版本的 doctype 而言&#xff0c;要简单明了得多&#xff1a; <!DOCTYPE html><html> ...创建一个名为 index.html 的文档&#xff0c;或者猛击这里下载这份…

软工读书笔记 week 1

这次读书笔记主要是就《程序员修炼之道》这本书的前半部分做一些总结以及发表一些自己的看法。 本书前面的一部分主要是一些程序员应该在工作中时刻注意的事情&#xff0c;一些关键的信息如下&#xff1a; 1、处理问题的态度与责任。 这是在本书序言中就提到的&#xff0c;可见…

mysql脚本解读_一篇很好的关于mysqld_safe脚本源码解读的文章,收藏了!!

#!/bin/sh# 一些状态变量的定义KILL_MYSQLD1; # 试图kill多余的mysqld_safe程序&#xff0c;1表示需要killMYSQLD# mysqld二进制可执行文件的名称niceness0# 进程的调度优先级标识# 下面的变量主要用于标识不使用错误日志和sysloglogginginit # 日志记录状态&#xff0c;init代…

Java EE 7发布–反馈和新闻报道

Java EE 7已经存在了几天。 我们所有人都有机会观看直播活动或可用的重播 。 最后的MR版本完成了将他们的工作推向JCP的过程&#xff0c;基本上是一个总结。 是时候反思发生的事情以及我对此的想法了。 启动活动中的社区参与 这不是一个大秘密。 即使Oracle的Java EE 7发行可以…

进制转换器

1 /*2 进制转换器 3 P进制数x转为Q进制数z4 过程模拟&#xff0c;先转十进制再转目标进制 5 */6 #include<cstdio> 7 #include<algorithm>8 #include<stack>9 using namespace std; 10 int main(){ 11 int p,q; 12 int x,y0,z; 13 int a1; 14 …

HTML5中的本地数据库-Web SQL Database

html5增加新的特性&#xff0c;那就是增加了本地存储&#xff01;改善用户体验&#xff0c;或许html5会带着我们走进新的互联网时代。 下面看看怎样操作web 数据库吧&#xff01;&#xff01;首先新建数据库&#xff01; var db window.openDatabase("mydata", &quo…

使用Apache CXF开发SOAP Web服务

在上一篇文章中&#xff0c;我逐步介绍了使用apache CXF开发简单的RESTFull服务的步骤。 在本文中&#xff0c;我将讨论使用CXF开发SOAP Web服务。 在继续前进之前&#xff0c;让我们了解构成SOAP Web服务的一些概念/元素。 SOAP或简单对象访问协议 SOAP是一种协议&#xff0c…

pkill mysql_每天一个linux命令:kill命令

终止一个进程或终止一个正在运行的程式&#xff0c;一般是通过kill 、killall、pkill、xkill 等进行。比如一个程式已死掉&#xff0c;但又不能退出&#xff0c;这时就应该考虑应用这些工具。另外应用的场合就是在服务器管理中&#xff0c;在不涉及数据库服务器程式的父进程的停…

HTML5中如何检查浏览器是否支持本地存储

如果你的浏览器支持该特性的话&#xff0c;那么全局对象&#xff1a;window上会有一个localStorage的属性&#xff0c;反之&#xff0c;你的浏览器不支持的话&#xff0c;那么该属性值为undefinedJavaScript function supports_local_storage(){ return !!window.localStorage…

解决swiper-slide在ion-slide-box不滑动的问题(暂且这么描述)

1&#xff0c;开发环境ionic1angularjs1 嗯对的版本都是最低版本 页面结构想要完成的功能是这样的&#xff08;比较丑 &#xff0c;不接受批评&#xff0c;捂脸&#xff09; 大致如图 代码结构&#xff1a; <ion-slide-box><ion-slide> //列表 <div class"…

java趣味题-打印99乘法表

java趣味题-打印99乘法表 public class First102 { public static void main(String[] args) { for(int i1;i<9;i){ for(int j1;j<i;j){ System.out.print(i"*"j""i*j" "); } System.out.println(""); } } }posted…

MOXy的@XmlVariableNode – JSON模式示例

我们正在向EclipseLink MOXy添加从域模型生成JSON模式的 功能 。 为此&#xff0c;我们创建了一个新的变量节点映射。 在本文中&#xff0c;我将通过将Java模型映射到JSON模式来演示新的映射。 您可以使用每晚构建的EclipseLink 2.6.0进行尝试&#xff1a; http://www.eclips…

在HTML5中如何利用Canvas处理并存储图片

HTML5中增加的Canvas元素&#xff0c;配合JS灵活的语法&#xff0c;处理起图片变得异常简单&#xff0c;不需要在客户端用C/C 写一大堆代码&#xff0c;对于熟悉JS的程序员来说&#xff0c;只需要考虑处理图片的逻辑了。 canvas中如果想要处理图片就需要借助ImageData这个对象&…

python正则表达式research_八Python正则表达式

正则表达式是搜索、替换和解析复杂字符串的一种强大而标准的方法&#xff0c;Python中的正则相关的东西全在re模块下。1 常用的匹配^匹配字符串的开始$匹配字符串的结尾\b匹配一个单词的边界\d匹配任意数字\D匹配任意非数字字符x?匹配一个可选的x(匹配1次或0次x字符)x*匹配0次…

Win10 系统直接在目录下打开cmd

每次用cmd命令&#xff0c;就要定位到当前文件夹&#xff0c;很麻烦&#xff0c;于是想了下&#xff0c;可不可以直接定位到要操作的文件夹&#xff0c;百度了一下&#xff0c;果然&#xff0c;度娘没有让我失望&#xff0c; 美滋滋的试了下。真的可以。在此记录下&#xff0c;…