测试驱动开发 测试前移_测试驱动陷阱,第2部分

测试驱动开发 测试前移

单元测试中单元的故事

在本文的上半部分 ,您可能会看到一些不好但很受欢迎的测试示例。 但是我不是一个专业的批评家(也被称为“巨魔”或“仇恨者”),没有任何建设性的话就抱怨。 多年的TDD教给我的不仅仅是事情会变得多么糟糕。 有许多简单但有效的技巧,可以使您的测试生活变得更加轻松。

想象一下:您有一家小公司中一间小型会议室的预订系统。 由于某些奇怪的原因,它必须处理离线预订。 人们将他们的预订请求发布到某个前端,每周一次,您会收到一个文本文件,其中包含公司的工作时间以及所有的预订(在当天,多长时间,由谁,在什么时间点提交)订购。 您的系统应根据一些业务规则(先到先得,仅在办公室工作时间内提供此类服务)为房间生成日历。

作为分析的一部分,我们提供了明确定义的输入数据和预期结果,并带有示例。 确实,TDD的情况很好。 可悲的是,在现实生活中从来没有发生过这样的事情。

我们的示例测试数据如下所示:

class TestData {static final String INPUT_FIRST_LINE = '0900 1730\n';static final String FIRST_BOOKING    = '2011-03-17 10:17:06 EMP001\n' +'2011-03-21 09:00 2\n';static final String SECOND_BOOKING   = '2011-03-16 12:34:56 EMP002\n' +'2011-03-21 09:00 2\n';static final String THIRD_BOOKING    = '2011-03-16 09:28:23 EMP003\n' +'2011-03-22 14:00 2\n';static final String FOURTH_BOOKING   = '2011-03-17 10:17:06 EMP004\n' +'2011-03-22 16:00 1\n';static final String FIFTH_BOOKING    = '2011-03-15 17:29:12 EMP005\n' +'2011-03-21 16:00 3';static final String INPUT_BOOKING_LINES =FIRST_BOOKING +SECOND_BOOKING +THIRD_BOOKING +FOURTH_BOOKING +FIFTH_BOOKING;static final String CORRECT_INPUT = INPUT_FIRST_LINE + INPUT_BOOKING_LINES;static final String CORRECT_OUTPUT = '2011-03-21\n' +'09:00 11:00 EMP002\n' +'2011-03-22\n' +'14:00 16:00 EMP003\n' +'16:00 17:00 EMP004\n' +'';
}

现在,我们从一个积极的测试开始:

BookingCalendarGenerator bookingCalendarGenerator =  new BookingCalendarGenerator();@Test
public void shouldPrepareBookingCalendar() {//whenString calendar = bookingCalendarGenerator.generate(TestData.CORRECT_INPUT);//thenassertEquals(TestData.CORRECT_OUTPUT, calendar);
}

看来我们已经使用“生成”方法设计了BookingCalendarGenerator。 很公平。 让我们添加更多测试。 测试业务规则。 我们得到这样的东西:

@Testpublic void noPartOfMeetingMayFallOutsideOfficeHours() {//givenString tooEarlyBooking = '2011-03-16 12:34:56 EMP002\n' +'2011-03-21 06:00 2\n';String tooLateBooking = '2011-03-16 12:34:56 EMP002\n' +'2011-03-21 20:00 2\n';//whenString calendar = bookingCalendarGenerator.generate(TestData.INPUT_FIRST_LINE + tooEarlyBooking + tooLateBooking);//thenassertTrue(calendar.isEmpty());}@Testpublic void meetingsMayNotOverlap() {//givenString firstMeeting = '2011-03-10 12:34:56 EMP002\n' +'2011-03-21 16:00 1\n';String secondMeeting = '2011-03-16 12:34:56 EMP002\n' +'2011-03-21 15:00 2\n';//whenString calendar = bookingCalendarGenerator.generate(TestData.INPUT_FIRST_LINE + firstMeeting + secondMeeting);//thenassertEquals('2011-03-21\n' +'16:00 17:00 EMP002\n', calendar);}@Testpublic void bookingsMustBeProcessedInSubmitOrder() {//givenString firstMeeting = '2011-03-17 12:34:56 EMP002\n' +'2011-03-21 16:00 1\n';String secondMeeting = '2011-03-16 12:34:56 EMP002\n' +'2011-03-21 15:00 2\n';//whenString calendar = bookingCalendarGenerator.generate(TestData.INPUT_FIRST_LINE + firstMeeting + secondMeeting);//thenassertEquals('2011-03-21\n15:00 17:00 EMP002\n', calendar);}@Testpublic void orderingOfBookingSubmissionShouldNotAffectOutcome() {//givenList<String> shuffledBookings = newArrayList(TestData.FIRST_BOOKING, TestData.SECOND_BOOKING,TestData.THIRD_BOOKING, TestData.FOURTH_BOOKING, TestData.FIFTH_BOOKING);shuffle(shuffledBookings);String inputBookingLines = Joiner.on('\n').join(shuffledBookings);//whenString calendar = bookingCalendarGenerator.generate(TestData.INPUT_FIRST_LINE + inputBookingLines);//thenassertEquals(TestData.CORRECT_OUTPUT, calendar);}

仅此而已。 但是,如果我们得到一些垃圾作为输入怎么办。 还是如果我们得到一个空字符串? 让我们为此设计:

@Test(expected = IllegalArgumentException.class)public void rubbishInputDataShouldEndWithException() {//whenString calendar = bookingCalendarGenerator.generate('rubbish');//then exception is thrown}@Test(expected = IllegalArgumentException.class)public void emptyInputDataShouldEndWithException() {//whenString calendar = bookingCalendarGenerator.generate('');//then exception is thrown}

IllegalArgumentException很公平。 我们不需要再花哨的方式处理它。 我们已经完成了。 最后,让我们在测试下编写该类:BookingCalendarGenerator。

因此,我们做到了。 结果表明,对于一个方法来说,整个过程有点大。 因此,我们使用了“提取方法”模式的强大功能。 我们将代码片段分为不同的方法。 我们将可操作的方法和数据分组为类。 我们使用面向对象编程的功能,使用单一职责原理,使用组合(准确地说是分解),最后得到一个像这样的程序包:

我们有一个公共类,和几个包作用域类。 这些包范围类显然属于公共类。 为了清晰起见,这是一个类图:

那些不是愚蠢的数据对象。 这些是成熟的课程。 具有行为,责任感和封装感。 这是我们测试驱动者可能想到的一件事:我们没有针对这些类的测试。 我们只有公共班级。 不好,对吧? 没有测试一定是不好的。 很坏。 对?

错误。

我们有测试。 我们启动了代码覆盖率工具,然后看到:100%的方法和类。 95%的线。 不错(在下一篇文章中,我将达到不确定性的5%)。

但是我们只有一个单元测试类。 这样好吗

好吧,让我强调一下,指出答案:

这是一个UNIT测试。 有理由将其称为UNIT测试!

该单元不必是一个单一的类。 该单元不必是单个包装。 由您决定单位。 这是一个通用名称,因为您的理智和常识应该告诉您停止的地方。

所以我们有六个班级作为一个单元,有什么大不了的? 除了其他人之外,是否有人要使用其中一个类别呢? 他不会对此进行测试,对吗?

错误。 除了在测试中实际调用的那些类之外,这些类都是包范围的。 这个包裹范围的事情告诉您:“退后一步。 不要碰我,我属于这个包裹。 不要试图单独使用我,我是设计在这里!”。

因此,是的,如果程序员将其中之一或将其公开,他可能会知道,所有保证都将失效。 写你自己的测试,伙计。

我被问到是否有人要向其中一个类添加某些行为呢? 他怎么会知道他没有破东西?

好吧,他将从测试开始,对吗? 是TDD,对不对? 如果需求有变化,则将此变化编码为测试,然后,直到那时,您才开始弄乱代码。 因此,您是安全的。

我看到人们盲目地写每堂课的测试,却没有思考,这让我哭了。 我最近做了很多配对编程,您知道我发现了什么吗? Java程序员通常不使用package-scope。 Java程序员通常不知道,这种保护意味着:对我来说,我的所有后代和每个人都在同一软件包中。 没错,受保护不仅仅是包范围,更不是一点点。 因此,如果Java程序员不知道包范围实际上是什么,并且与Groovy相反,这是默认值,那么他们如何理解单元是什么?

我能得到多高?

现在,有一个有趣的想法:如果我们可以对一个包进行单个测试,那么我们可以对一个包树进行单个测试。 你知道,像这样:

我们都知道Java中的软件包不是真正的树状包,唯一具有目录结构的是按照非常古老的约定,而且我们知道目录结构仅用于解决名称冲突问题,尽管如此,我们还是倾向于使用包,就像name.after.the.dot有意义一样。 就像我们可以将一个包裹隐藏在另一个包裹中一样。 或与他们建立千层面的烤宽面条。

那么可以为一个树形树使用一个测试类吗?

是的。

但是,如果是这样,到哪里结束? 我们可以从包树一直到应用程序的入口点吗? 这些……可能是集成测试或功能测试。 我们能做到吗? 那会好吗?

答案是:可以。 在一个完美的世界中,那会很好。 在我们那肮脏的,悬挂在刀刃上的世界里,这简直是疯了。 为什么? 因为功能性的端到端测试很慢。 太慢了。 太慢了,以至于您想将它们扔掉,然后到某个地方,而不必总是等待某些东西。 一个充满创造力,不断反馈和闪电般快速安全的地方。

您将返回单元测试。

甚至还有更多原因。 一种是,很难对应用程序的所有流程进行端到端测试。 您可能应该对所有主要流程都执行此操作,但是对于错误,连接不良以及所有可能在一处或另一处抛出的棘手逻辑部分呢? 不,有时候像这样设置集成测试环境会太困难了,所以最终还是要用单元测试来测试它。

第二个原因是,尽管功能测试并未在代码上浇注具体的内容,但不会通过在测试用例中重复执行算法来抑制您的创造力,但是它们也没有提供重构的安全性。 当您拥有一个具有单个公共类的程序包时,很明显有人可以安全地做什么,而他不能做什么。 当您将某些内容包含在库或插件中时,它仍然很明显。 但是,如果您有成千上万个公共类,并且要实现一个新功能,则可能要使用其中一些,并且您想知道它们很好。

因此,不,在我们的世界中,仅进行功能测试是没有意义的。 抱歉。 但是按类创建测试也没有意义。 出于某种原因,它被称为UNIT测试。 用那个

祝您编程愉快,别忘了分享!

参考: 测试驱动陷阱,来自我们JCG合作伙伴 Jakub Nabrdalik的第2部分 ,在Solid Craft博客上。


翻译自: https://www.javacodegeeks.com/2012/09/test-driven-traps-part-2.html

测试驱动开发 测试前移

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

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

相关文章

CentOS6.3 重启后/etc/resolv.conf 被还原解决办法

今天一台服务器上不了网&#xff0c;设置了nameserver&#xff0c;重启后/etc/resolv.conf文件就被自动还原了&#xff0c;最后发现是被Network Manager修改了。 解决方法&#xff1a; 停止Network Manager服务 service NetworkManager stop重启网络服务 /etc/init.d/network r…

react安装_超全面详细一条龙教程!从零搭建React项目全家桶(上篇)

React是近几年来前端项目开发非常火的一个框架&#xff0c;其背景是Facebook团队的技术支持&#xff0c;市场占有率也很高。很多初学者纠结一开始是学react还是vue。个人觉得&#xff0c;有时间的话&#xff0c;最好两个都掌握一下。从学习难度上来说&#xff0c;react要比vue稍…

MATLAB中find函数详解

&#xff08;转自http://blog.sina.com.cn/emily250886&#xff09; 功能&#xff1a; 寻找非零元素的索引和值语法&#xff1a;1. ind find(X) 2. ind find(X, k) 3. ind find(X, k, first) 4. ind find(X, k, last) 5. [row,col] find(X, ...) 6. [row,col,v] f…

Oracle Service Bus简介

我们正在为电信提供商设计一个新系统&#xff0c;在该系统中&#xff0c;我们研究了用作企业服务总线的Oracle服务总线&#xff08;OSB&#xff09;。 对我来说&#xff0c;第一个优点是它提供了惊人的工具支持。 通过将其捆绑为Oracle SOA套件&#xff0c;Oracle已将其所有企业…

VS2008中源文件与模块生成时的文件不同,仍要让调试器使用它吗

解决办法: 1.在工具-> 选项-> 调试-> 常规 里 把“要求源文件和版本匹配”的勾勾去掉 然后去掉“地址级的调试”的勾勾&#xff08;这样就不会出现反汇编代码了&#xff09; 然后重新引用 2.直接把WEB项目下面的Bin文件夹下的BLL全部删除。 3.若出现如下的警告信息 仔…

Linux中的15个‘echo’ 命令实例

作者&#xff1a; tecmint Avishek Kumar 译者&#xff1a; LCTT geekpi | 2014-10-03 12:42 评论: 4 收藏: 5 分享: 5 echo是一种最常用的与广泛使用的内置于Linux的bash和C shell的命令&#xff0c;通常用在脚本语言和批处理文件中来在标准输出或者文件中显示一行文本或…

springboot redis 断线重连_Redis高可用方案实现

redis中为了实现高可用&#xff08;High Availability&#xff0c;简称HA&#xff09;&#xff0c;采用了如下两个方式&#xff1a;主从复制数据。采用哨兵监控数据节点的运行情况&#xff0c;一旦主节点出现问题由从节点顶上继续进行服务。主从复制redis中主从节点复制数据有全…

罗德里格斯(Rodrigues)旋转向量与矩阵的变换

在做双目立体视觉深度图像生成的时候&#xff0c;遇到旋转向量&#xff08;1x3&#xff09;与旋转矩阵&#xff08;3x3&#xff09;的概念&#xff0c;得知二者可以通过罗德里格斯相互转化。 1.旋转的表示 处理三维旋转问题时&#xff0c;通常采用旋转矩阵的方式来描述旋转变换…

Mongodb 分片与副本集

测试搭建192.168.3.110mongos 30000&#xff0c;30001,30002config 40000,40001,40002shard1 50001,50002,50003shard2 50004,50005,50006,shard3 50007,50008,50009[rootmysql-slave10 data]# mkdir -p /data/config/1/data[rootmysql-slave10 data]# mkdir -p /data/config/2…

python必读5本书籍_免费下载!5本从Python入手机器学习的必备电子书!(附链接)...

大数据文摘授权转载自数据派THU作者&#xff1a;Richard Gall编译&#xff1a;车前子、丁楠雅今天&#xff0c;机器学习已成为软件工程所有领域中最重要的发展趋势之一。这门技术不再局限于研究者和数据分析者&#xff0c;从网络安全到网络开发机&#xff0c;器学习都至关重要。…

error c4996: 'fopen' This function or variable may be unsafe如何解决

1.问题描述 VS2013OPENCV249&#xff0c;不止一次遇到以下类似错误&#xff1a; error C4996: fopen: This function or variable may be unsafe. Consider using fopen_s instead. To disable deprecation, use _CRT_SECURE_NO_WARNINGS. See online help for details.d:\open…

JavaParser中AST节点的观察者

我们离JavaParser 3.0的第一个候选发布版本越来越近。 我们添加的最后一项功能是支持观察抽象语法树的所有节点的更改。 当我为此功能编写代码时&#xff0c;我收到了Danny van Bruggen&#xff08;又名Matozoid&#xff09;和Cruz Maximilien的宝贵反馈。 因此&#xff0c;我使…

nc命令详解

NetCat&#xff0c;在网络工具中有“瑞士军刀”美誉&#xff0c;其有Windows和Linux的版本。因为它短小精悍&#xff08;1.84版本也不过25k&#xff0c;旧版本或缩减版甚至更小&#xff09;、功能实用&#xff0c;被设计为一个简单、可靠的网络工具&#xff0c;可通过TCP或UDP协…

startindex 不能大于字符串长度_「12」学习MySQL第二类函数:字符串函数

前一篇文章总结了下MySQL中常用的数学函数&#xff0c;本文接下来的重点就是认识“字符串函数”。上图这匹漂亮的马儿就是通过各种字符(字母、括号、逗号、竖线等)画出来的。我不由自主地会去想下面几个问题&#xff1a;它到底有多少个字符&#xff1f;--字符串长度它里面的小写…

对数函数的C、MATLAB表示

1、C语言中的对数函数 头文件&#xff1a;#include格式&#xff1a;①double log (double x);log() 函数返回以 e 为底的对数值&#xff0c;即数学中的lnx&#xff1b;如果 x 为负数或 0&#xff0c;则会发生错误并设置 errno 值。错误代码&#xff1a;EDOM&#xff1a;参数x 为…

微信小程序开发——超链接或按钮点击跳转到其他页面失效

1. 超链接导航失效&#xff1a; 小程序规则——wx.navigateTo 和 wx.redirectTo 不允许跳转到 tabbar 页面&#xff0c;只能用 wx.switchTab 跳转到 tabbar 页面转载于:https://www.cnblogs.com/xyyt/p/9252835.html

javaone_JavaOne 2012:Lambda之路

javaone我最热切期待的JavaOne 2012演讲之一是Brian Goetz的“通往Lambda的道路”。 昨晚的技术主题演讲中的Lambda味道仅增加了预期。 这是在希尔顿广场A / B举行的&#xff0c;距离我之前在金门大桥A / B / C参加的演讲仅几步之遥。 我曾预计会打包相对较大的Plaza A / B&…

uml 时序图_UML各种图总结:

UML的书籍&#xff1a;《UML distilled》、《UML和模式应用》、《UML用户指南》、《UML对象设计与编程》、《UMLOracle8建模》UML&#xff08;Unified Modeling Language&#xff09;是一种统一建模语言&#xff0c;为面向对象开发系统的产品进行说明、可视化、和编制文档的一种…

让程序结果在屏幕上暂停一段时间

现在使用VS2013输出printf显示结果&#xff0c;经常会出现一闪而过的情况&#xff0c;现有以下解决办法&#xff1a; 1.调用系统命令暂停 #include <stdlib.h>system("pause");2. #include<conio.h> getchar(); 3. #include<stdlib.h> getchar(); …

Linux shell的标准输入、输出和错误

1.文件描述符 在linux shell执行命令时&#xff0c;每个进程都和三个打开的文件相联系&#xff0c;并使用文件描述符来引用这些文件。由于文件描述符不容易记忆&#xff0c;shell同时也给出了相应的文件名&#xff1a; 文件文件描述符输入文件—标准输入0&#xff08;缺省是键…