- 浏览: 119388 次
- 来自: ...
文章分类
最新评论
命名
测试用例的名字应该描述需求, 不要描述实现.
取决于你要沟通交流传递的信息, Test Case 有至少两个作用
-
检查你的产品代码是否按预期工作, 这由函数体来完成
-
表达你的预期,让阅读代码的人知道你的产品能够干什么,如何使用, 甚至如何设计的;这除了函数体的assert语句外,Test case的名字更是重要的手段
但我们通常只会为一段测试代码起一个名字, 而要表达的信息如此之多, 怎么办? 一个测试用例尽量只有一个断言, 或至少限制在一类断言, 这时候你就能够起一个Domain相关的名字
如果Domain比较复杂, 进入项目的新人不一定了解, 这时候你的测试用例的名字就是最好的领域知识, 比如电信计费系统:
public void testShouldBeFreeFrom2amTo5am() throws Exception {
Duration talkingDuration = new Duration("2am", "5am");
Money fee = chargeSystem.charge(talkingDuration);
assertEquals(Money.ZERO, fee);
}
Setup
所有测试相关的代码, 包括设置测试环境, 调用被测对象触发测试, 断言测试结果等, 应该放在一起,便于阅读和理解; 那setUp()里放什么?
-
如果一个对象很容易用一两句话装配, 一般可以在每个测试用例里就近创建它
-
如果初始化一个对象很复杂, 要不少代码, 就写一个函数来做初始化, 然后在每个测试用例里就近调用它
-
setUp()里放点公用的基础的比如对外部依赖环境的设置
Mock stub
1, 真实数据/环境
随着自动化单元测试, Mock技术的流行, 人们似乎逐渐的鄙视使用真实的环境和数据来测试; 但真实的环境和数据并不是天然就与自动化无缘, 只要它可以很容易的获得,并能够伴随测试代码一起发布
这方面典型例子是文件系统, 可以使用相对路径消除对存放位置的依赖, 使用统一的入口如 TestFilesUtil 来负责对测试数据的访问.
真实数据/环境的好处是它最接近生产环境, 可以通过提供不同的数据来产生不同的测试用例, 比较方便的提高测试覆盖率
真实数据/环境的坏处就是应用范围比较窄,虽然有"嵌入式"的数据库和FTP服务器等,但很少在单元测试中使用;我们主要是在测试以文件作为输入的API的时候使用真实的文件系统
2, 静态手写Stub
这是初期比较常用的方法, 是State Based Test方法的实现方式 (与之对应的Interaction Based Test/ Behaviour verification Test的实现方式是Mock Object, 或者Classical TDD vs. Mocklist TDD, 反正就是这么两个意思
好处是简单, 易于获得, 符合既有的思维习惯, 坏处是繁琐, 最终测试代码中充斥着大量Stub类
Stub类的编写应该遵循以下原则:
-
不要包含任何逻辑;
就是所有函数都简单的return一个固定的结果; 一旦你的Stub包含逻辑, 你就需要为你的Stub也写一个测试了, 呵呵, 我们的系统中现在就有这么一个Stub, 陪伴它的是它的测试用例
Stub 会带来一个好处:
-
强迫你重新考虑你的设计 (其实这几乎是任何高质量的测试用例都能带来的)
但Stub尤其会在两个方面强迫你重新思考:
1), 调用链
就是当你的产品代码中出现如下调用时 DateTime buildDate = obj.getConfigure().getProject().getBuild().getDate();
你如何使用Stub来测试这段代码呢? 很不幸, 你需要一鼓作气写三个Stub类分别代理Configure/Project/Build才能完成测试; 这就强迫你重新思考, 原来的设计是否有问题, 是否需要重构
2), 针对接口编程
OO完美主义者倾向于用构造函数建立起一个不变式, 所以经常在构造函数中进行各种计算和验证, 一旦发现不符合不变式就会抛出异常之类; 这就给Stub带来一个问题, 因为要调用基类的构造函数, 传递什么样的参数才能不让基类的构造函数抛出异常呢? 通常在测试环境中我们不容易满足这类约束, 这就迫使我们抽取本来就应该抽取的接口, 将函数的参数类型或返回值类型替换为该接口, 然后只要Stub实现这个接口即可
3, Mock Object
真实数据是测试世界的北极, Mock Object就是南极
Mock Object的经验不多, 所以首先感受到的是它的不便之处: 重构会破坏测试用例, 即使你的重构是正确的
与有的Mock Object实现对重构的"Rename"之类的操作支持不好相比, 重构直接破坏测试用例更郁闷:
-
比如你用一个现有的充分测试过的第三方的API替换了你自己写的几行实现
-
比如你删除了几行冗余的调用
-
甚至你把几个public函数中重复逻辑抽取到一个私有函数里, 都有可能破坏基于Mock Object的test case.
然而与Mock Object的误用相比, 重构破坏测试用例便不算什么了. 毫无疑问有些情况下是不应该使用Mock Object的, 这尤其体现在那些紧凑的API上, 即单一API调用, 根据不同参数返回不同结果; 这类API包括:
-
传入一个xpath表达式, 返回NodeList
-
传入正则表达式, 返回匹配结果
-
传入SQL语句, 返回结果集
这类API的返回结果强烈依赖于它们的参数, 参数才是它们的核心, 你一旦在你的产品代码中使用了这些API, 又在测试用例中Mock了这些API, 直接返回固定的结果, 那么恭喜你, 你的测试白做了. (这类API对Stub也不感冒, 尽可能用真实数据来测)
Mock Object的适用场景其实和Stub差不多, 首要目的是减少对系统其它部分包括外部系统的依赖. 但Mock对交互顺序和参数/返回值传递强大的支持可以使你更精确的断言你的代码的行为 考虑经典的用户存款场景, 假设在此过程中, 你的API会进一步调用银行API来完成操作, 如果使用State Based 测试方法, 我们的测试用例可能只是在调用你的API前后断言一下用户账户的balance就算了, 比如:
public void testBalanceShouldIncrease50WhenDeposit50() throws Exception {
double balanceBeforeDeposit = getBalance();
deposit(50);
double balanceAfterDeposit = getBalance();
assertEquals(balanceBeforeDeposit + 50, balanceAfterDeposit);
}
通常这样的测试也算测过了, 但这样我们无法测试钱是直接进了你的账户., 还是中间流转了一下; 如果金融系统的客户对自己内部的系统要求比较严格, 你可能需要对这中间的内部调用逻辑进行测试, 以避免洗钱之类的可能, 这时候你就可以用Mock来断言银行的API确实以参数50被调用了一次, 而不是以参数25调用了两次
public void testDepositShouldPutMoneyToYourAccountDirectly() throws Exception {
depositMock.expect(once()).method("deposit").with(eq(50));
double balanceBeforeDeposit = getBalance();
deposit(50);
double balanceAfterDeposit = getBalance();
assertEquals(balanceBeforeDeposit + 50, balanceAfterDeposit);
}
Mock Object 还支持交互顺序的测试. 比如你有两个操作, 必须以确定的顺序来执行, 你担心后续的维护者会破坏这种约定, 则可以使用 Mock 测试显式的描述它.
关于 Mock 和 Stub 的其它描述, 请参考 <<敏捷质疑: TDD>>
测试杀手
1.
static
method, operator
这是一个Spring的时代了, 你还在用
static
method 吗? 还在用 operator 去创建对象吗? 仅仅"让你的API难以测试"这一条便足以宣判他们的死刑了
记得很早以前就写过: "
RAII让我告别了delete,IoC让我告别了
"
关于 static和
new 的测试, 请参考<<假冒的艺术>>
2. Prolonged failed test case
"小步前进"是确保TDD成功的众多因素中的一条, 更早以前还写过: "
目标驱动生活,每天早上运行一遍测试用例:assertTrue(有房);assertTrue(有车),测试失败,就努力让它早日通过
"
感谢当年提出"步子太大,这个测试fail的时间太长"意见的朋友,Blog的变迁使得这条评论已经不见了, 但真理亘古不变,我现在决定把它改成"assertTrue(每月咱也打回的); assertTrue(不要顿顿三明治)"
测试朋友
1, 谁是谁的测试,
当你重构测试用例的时候?
当你重构产品代码的时候, 测试用例是你的朋友, 可以确保重构不会引入错误;
那么当你的测试越来越多, 需要重构一下便于增加新的测试的时候, 谁是谁的测试呢?
2. IDE
我的设想是强迫你TDD的IDE; 现在的IDE, 如Eclipse, 它的新建Class的向导不会让你选择这个新类需要满足的测试用例, 而它的新建测试用例向导会让你选择打算测试哪个类
新的支持TDD的IDE会反过来, 你不能凭空新建一个类, 除非它是测试用例类, 当你新建类的时候, 你必须指定它被用来满足哪个失败的Test Case
3.评审员
直接从以前的Blog中搬过来: "测试一词是个错误的用法,表达了一种实现,手段,过程,而不是目标、结果;一个副作用就是令测试人员不被重视;代码复审和系统测试目的都是找出编码中的错误,但一种叫评审员,一种叫测试人员;建议在目前出现测试一词的地方,都替换为“验证”,如系统验证,验证人员,“xxx,那个bug我改了,你再帮我验证一下”,呵呵,颠覆一把话语体系"
发表评论
-
Architecture is layered
2004-12-11 11:57 379那天被问道软件架构师需要了解编程语言的细节吗? 呵呵,架构是 ... -
Thinking Everyday
2004-12-11 12:01 4401,编程语言的发展趋势 ... -
糟糕命名集锦
2004-12-11 16:50 5751,公交支线,如375和375 ... -
古代的软件开发 (一)
2005-02-19 16:45 6781,额外的中间层鞋子:人类发明鞋子的意义无论如何评价都不过分, ... -
访问控制 : 语言和平台
2005-03-15 19:27 616程序逻辑上的组织方式(如名称空间,包等)可以和部署时的分发 ... -
Thinking Everyday II
2005-03-17 15:11 6211, 是业务,不是技术,傻瓜 是集成,不是编程 是使用,不 ... -
内容与标准为王:下一代互联网与下一代搜索
2005-07-25 14:53 710第一代互联网混淆了真正的数据和它的表现形式,第一代搜索无法 ... -
个性与定制为王:下一代互联网和下一代门户
2005-07-28 11:28 613看一下现在我与互联网有关的生活:我有两三个常用的Web邮箱 ... -
泛型编程 vs. 面向对象
2005-08-10 14:30 836面向对象:封装(数据抽象)是基础,继承是手段,多态是目的 ... -
函数式编程 vs. 对象式编程
2005-08-10 14:44 651<<我爱我家>>有一集和平摔成了脑 ... -
用手机从ATM取钱
2005-11-21 22:49 697手机的以下两个特性,使它潜在的可能成为统一的支付和信用平 ... -
Web 3.0 : Unified Human-like Interaction
2006-01-14 16:31 700你还在到搜索引擎的主页上去搜索吗?你还登录新闻网站查询最新比赛 ... -
软件生物学
2006-01-14 16:59 651长久以来,软件的建筑学隐喻已经深入人心,可始终无法达到建筑 ... -
广义对象论
2006-01-25 15:31 714前几天本想接着以前的思维中对“3.2 Programming ... -
Thinking Everyday III
2006-03-26 14:17 7921, RAII让我告别了delete,IoC让我告别了ne ... -
简单至及的AOP和IOC
2006-03-26 14:21 664I. AOP的例子 1, Google To ... -
Thinking Everyday IV
2007-05-15 04:36 5291, 实际上 C# 2.0 已经部 ... -
迭代本质论
2008-02-14 13:58 633新年伊始, 可能你又要制定一些计划了, 实际上, 你的生活在开 ... -
建筑的永恒之道
2004-08-10 18:31 6542,质 这种特质是任 ... -
姑苏慕容与软件开发
2004-10-06 11:47 635一,逆向工程 那女子 ...
相关推荐
1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、本项目仅用作交流学习参考,请切勿用于商业用途。
DATA: 可供参考的微博评论数据。 详见 /DATA/weibocommennts.csv CODE: apiGetSheet.py 调用百度API 获得 微博评论对应 文字的 情感得分, sheetGetvalue.py 根据情感得分进行标准化,获得实际倾向。
管理员 管理员信息管理 负责人管理 员工信息管理 公告信息管理 小型车收费标准设置(元/每公里) 大卡车收费标准设置(元/吨公里) 收费信息统计,统计小车和卡车收费,按月统计 负责人 个人资料修改 公告查看 小车收费统计(某员工某月统计) 大卡车收费统计(某员工某月统计) 员工 个人资料修改 公告查看 小型车收费登记(车牌号,车辆照片,行使公里数,收费金额,收费日期,收费员,按公里数可以自动计算费用 收费金额=收费标准*公里数) 大卡车金额设置(每吨/元)(车牌号,车辆照片,行使公里数,吨,收费金额,收费日期,收费员, 收费金额=收费标准*吨*公里数 ) 环境说明: 开发语言:Java 框架:ssm,mybatis JDK版本:JDK1.8 数据库:mysql 5.7 数据库工具:Navicat11 开发软件:eclipse/idea Maven包:Maven3.3 服务器:tomcat7
本套zabbix是基于6.0版本部署,内容涵盖zabbix的简介、zabbix server安装、zabbix基本概念、快速入门、zabbix进阶、zabbix实践、zabbix的高级监控使用。 =======知识领域 网络监控:监控网络设备、服务器和应用程序的运行状态。 系统监控:监控服务器性能,如CPU使用率、内存使用情况、网络流量等。 数据库监控:监控数据库性能和状态。 应用程序监控:监控应用程序的运行情况和性能指标。 云监控:监控云服务和虚拟机的健康状况
商品库存管理系统课程设计报告.docx
嘉兴智能卫浴项目建议书.docx
Java系统源码+夕阳红公寓管理系统 内容概要: 本资源包含了完整的Java前后端源码及说明文档,适用于想要快速搭建并部署Java Web应用程序的开发者、学习者。 技术栈: 后端:Java生态系统,包含Spring Boot、Shiro、MyBatis等,数据库使用Mysql 前端:Vue、Bootstrap、Jquery等 适用场景示例: 1、毕业生希望快速启动一个新的Java Web应用程序。 2、团队寻找一个稳定的模板来加速产品开发周期。 3、教育机构或个人学习者用于教学目的或自学练习。 4、创业公司需要一个可以立即投入使用的MVP(最小可行产品)。
(1)用户管理: 用户注册:新用户注册 用户登录:新用户登录 找回密码:忘记密码找回密码 用户评论:发表对新闻的评论 (2)管理员管理: 系统账号管理:管理员管理系统现有账号,进行删除停用等操作 系统公告管理:系统公告的发布和删除 新闻栏目管理:新闻栏目的新增和删除 ...
Java系统源码+计算机学院校友网 内容概要: 本资源包含了完整的Java前后端源码及说明文档,适用于想要快速搭建并部署Java Web应用程序的开发者、学习者。 技术栈: 后端:Java生态系统,包含Spring Boot、Shiro、MyBatis等,数据库使用Mysql 前端:Vue、Bootstrap、Jquery等 适用场景示例: 1、毕业生希望快速启动一个新的Java Web应用程序。 2、团队寻找一个稳定的模板来加速产品开发周期。 3、教育机构或个人学习者用于教学目的或自学练习。 4、创业公司需要一个可以立即投入使用的MVP(最小可行产品)。
四六级报名管理系统.pdf
C语言PTA-数组答案代码合集(湖工1-50)
YOLOv8 安全背心识别项目代码 项目详细介绍请看链接: https://blog.csdn.net/qq_53332949/article/details/144543625 数据集详细介绍请看:https://blog.csdn.net/qq_53332949/article/details/141503406 数据集下载请看:https://download.csdn.net/download/qq_53332949/89711610?spm=1001.2101.3001.9500 按文件中requirements.txt文件配置环境即可使用。
后后勤智能1.。1. 1. 1. 管理系统-...pdf后勤智能1.。1. 1. 管理系统-...pdf后勤智能1.。1. 管理系统-...pdf后勤智能1.。管理系统-...pdf勤智能管理系统-...pdf
瓶子、塑料袋检测70-YOLO(v5至v9)、COCO、CreateML、Darknet、Paligemma、TFRecord、VOC数据集合集.rarset1(拍摄照片)-V15 2023-08-09 3:43 PM ============================= *与您的团队在计算机视觉项目上合作 *收集和组织图像 *了解和搜索非结构化图像数据 *注释,创建数据集 *导出,训练和部署计算机视觉模型 *使用主动学习随着时间的推移改善数据集 对于最先进的计算机视觉培训笔记本,您可以与此数据集一起使用 该数据集包括133张图像。 汽车以可可格式注释。 将以下预处理应用于每个图像: *像素数据的自动取向(带有Exif-Arientation剥离) *调整大小为640x640(拉伸) 应用以下扩展来创建每个源图像的3个版本: * -24和+24度之间的随机旋转
管理员 个人密码修改 项目经理管理 员工管理(调试员,解决方案人员) 日志管理(用户登录日志) 系统设置 项目经理 个人信息管理,修改 项目模块管理 按项目分配调试员 调试员 个人信息管理,修改 BUG信息管理(各个环节的跟踪信息录入) 查看解决方案 解决方案人员 个人信息管理,修改 查看缺陷信息 提出解决方案 环境说明: 开发语言:Java 框架:ssm,mybatis JDK版本:JDK1.8 数据库:mysql 5.7 数据库工具:Navicat11 开发软件:eclipse/idea Maven包:Maven3.3 服务器:tomcat7
电路电压检测14-YOLO(v5至v9)、COCO、CreateML、Paligemma、VOC数据集合集.rar电路电压-V2释放 ============================= *与您的团队在计算机视觉项目上合作 *收集和组织图像 *了解非结构化图像数据 *注释,创建数据集 *导出,训练和部署计算机视觉模型 *使用主动学习随着时间的推移改善数据集 它包括132张图像。 电路电压以可可格式注释。 将以下预处理应用于每个图像: 没有应用图像增强技术。
项目包含前后台完整源码。 项目都经过严格调试,确保可以运行! 具体项目介绍可查看博主文章或私聊获取 助力学习实践,提升编程技能,快来获取这份宝贵的资源吧!
系统实现: 用户功能模块:用户点击进入到系统操作界面,可以对主页、个人中心、我的收藏管理、订单管理等功能模块,我的收藏管理:通过列表可以获取用户ID、收藏ID、表名、收藏名称、收藏图片信息并进行修改操作 管理员功能模块:管理员通过用户名和密码填写完成后进行登录。管理员登录成功后进入到系统操作界面,可以对主页、个人中心、用户管理、商品分类管理、商品信息管理、系统管理、订单管理等功能模块进行相对应操作。 项目包含完整前后端源码和数据库文件 环境说明: 开发语言:Java 框架:ssm,mybatis JDK版本:JDK1.8 数据库:mysql 5.7 数据库工具:Navicat11 开发软件:eclipse/idea Maven包:Maven3.3 服务器:tomcat7
Python课程设计,含有代码注释,新手也可看懂。毕业设计、期末大作业、课程设计、高分必看,下载下来,简单部署,就可以使用。 包含:项目源码、数据库脚本、软件工具等,该项目可以作为毕设、课程设计使用,前后端代码都在里面。 该系统功能完善、界面美观、操作简单、功能齐全、管理便捷,具有很高的实际应用价值。
Android 毕业设计,Android 毕业设计,小Android 程设计,含有代码注释,新手也可看懂。毕业设计、期末大作业、课程设计、高分必看,下载下来,简单部署,就可以使用。 包含:项目源码、数据库脚本、软件工具等,该项目可以作为毕设、课程设计使用,前后端代码都在里面。 该系统功能完善、界面美观、操作简单、功能齐全、管理便捷,具有很高的实际应用价值。