介绍

使用测试驱动开发大半年了,我还是对Stub和Mock的认识比较模糊,没有进行系统整理。

今天查阅了相关资料,觉得写得很不错,所以我试图在博文中对资料进行整理一下,再加上一些自己的观点。

本文是目前我对Stub和Mock的认识,难免有偏差,欢迎大家拍砖。

分析

Stub和Mock都是属于测试替身,对类型细分的话可以分为:

  • Dummy Object
  • Fake Object
  • Test Stub
  • Test Spy
  • Mock Object

前四项属于Stub,最后的Mock Object属于Mock。

类型分析

Dummy Object(哑对象)

测试代码仅仅是需要使用它来通过编译,实际上用不到它。如测试A类的run方法,需要在创建A类的实例时需要传入B类实例,但run方法并没有用到B类实例。在测试时需要传入B类的哑对象new NullB()(如“new A(new NullB())”),让其通过编译。这里的NullB是一个空类,没有具体实现。

Fake Object(假对象)

假对象相对于哑对象来说,要对耦合的组件有一些简单的实现,实现我们在测试中要用到的方法,指定期望的行为(如返回期望的值)。假对象适用于替换产品代码中使用的全局对象,或者创建的类。这里注意的是要先对被替换的全局对象或类进行备份,然后在测试完成后进行恢复。

示例1(替换全局对象):

//产品代码
function A(){
this.num = ;
}
A.prototype.run = function(){
this.num = window.b.getNum();
}; //测试代码
describe("测试A类的run方法", function(){
var temp = null; function backUp(){
window.b = window.b || {};
temp = YYC.Tool.extendDeep(window.b);
}
function restore(){
window.b = temp;
} beforeEach(function(){
backUp();
});
afterEach(function(){
restore();
}); it("获得数字", function () {
window.b = { //假对象
getNum: function(){
return ;
}
} var a = new A();
a.run(); expect(a.num).toEqual();
});
});

示例2(替换类):

//产品代码
function A() {
this.num = ;
this._b = new B();
}
A.prototype.run = function () {
this.num = this._b.getNum();
}; //测试代码
describe("测试A类的run方法", function () {
var temp = null; function backUp() {
window.B = window.B || function () {};
temp = B;
} function restore() {
window.B = temp;
} beforeEach(function () {
backUp();
});
afterEach(function () {
restore();
}); it("获得数字", function () {
window.B = function () {
};
window.B.prototype.getNum = function () {
return ;
}; var a = new A();
a.run(); expect(a.num).toEqual();
});
});

Test Stub(测试桩)

测试桩与假对象有点类似,也要实现与产品代码耦合的组件,指定期望的行为。这里最大的不同是测试桩需要注入到产品代码中,从而在测试产品代码时替换组件,执行桩的行为。使用测试桩不需要进行备份和还原。

示例:

//产品代码
function A(b) {
this.num = ;
this._b = b;
}
A.prototype.run = function () {
this.num = this._b.getNum();
}; //测试代码
describe("测试A类的run方法", function () {
it("获得数字", function () {
var stub_B = { //B类的桩
getNum: function(){
return ;
}
}; var a = new A(stub_B); //注入桩
a.run(); expect(a.num).toEqual();
});
});

Test Spy(嗅探桩)

与测试桩类似,但是可以记录桩使用的记录,并进行验证。

示例:

可以使用jasmine的spy来举例。

//产品代码
function A(b) {
this.num = ;
this._b = b;
}
A.prototype.run = function () {
this.num = this._b.getNum();
}; //测试代码
describe("测试A类的run方法", function () {
it("获得数字", function () {
var stub_b = {
getNum: function(){
return ;
}
};
spyOn(stub_b, "getNum").andCallThrough(); //嗅探桩的getNum方法 var a = new A(stub_b); //注入桩
a.run(); expect(a.num).toEqual();
expect(stub_b.getNum).toHaveBeenCalled(); //验证调用过桩的getNum方法
});
});

Mock Object(模拟对象)

设定产品代码中耦合的类的期望的行为,然后验证期望的行为是否发生,从而达到测试产品代码行为的目的。适用于验证一些void的行为。例如:在某个条件发生时,要记录Log。这种情景,用stub就很难验证,因为对目标物件来说,沒有回传值,也沒有状态变化,就只能通过mock object來验证目标物件是否正确的与Log介面进行互动。

示例:

//产品代码
function A(b) {
this.num = ;
this._b = b;
}
A.prototype.run = function () {
this.num = this._b.getNum();
}; //测试代码(Mock为伪代码)
describe("测试A类的run方法", function () {
it("获得数字", function () {
var mockB = Mock.createMock({
getNum: function(){}
}); //如果B类存在的话,也可以直接传入B的原型:var mockB = Mock.createMock(B.prototype);
Mock.expect(mockB.getNum, ).return().times(); var a = new A(mockB);
a.run(); expect(a.num).toEqual();
Mock.verify(); //验证期望的行为发生:mockB的getNum传入的参数为2;调用了1次mockB.getNum
});
});

Mock(Mock Object)与Spy(Test Spy)的比较

相同点

  • 都要注入到产品代码中。

不同的

  • Mock是替换整个被Mock的类,这个类可以存在也可以不存在。而Spy是使用一个已经存在的类,嗅探其中的部分方法。
  • 从流程中来说,Mock是先设定被Mock的类的期望行为,然后验证期望的行为是否发生。Spy是记录下桩的方法的使用记录(如传入的参数,调用的次数等),然后再对记录进行验证。

Mock退化为Stub

在现实使用中,我们经常将mock做不同程度的退化,从而使得mock对象在某些程度上如stub一样工作。
使用Mock的示例:
//产品代码
function A(b) {
this.num = ;
this._b = b;
}
A.prototype.run = function () {
this.num = this._b.getNum();
}; //测试代码(Mock为伪代码)
describe("测试A类的run方法", function () {
it("获得数字", function () {
var mockB = Mock.createMock({
getNum: function(){}
}); //如果B类存在的话,也可以直接传入B的原型:var mockB = Mock.createMock(B.prototype);
Mock.expect(mockB.getNum).return(); //只指定返回值,没有期望的参数或期望调用的次数。因此不用verify来验证了! var a = new A(mockB);
a.run(); expect(a.num).toEqual();
});
});

也可以用Stub来达到相同的效果:

//产品代码
function A(b) {
this.num = ;
this._b = b;
}
A.prototype.run = function () {
this.num = this._b.getNum();
}; //测试代码
describe("测试A类的run方法", function () {
it("获得数字", function () {
var stub_B = {
getNum: function(){
return ;
}
}; var a = new A(stub_B);
a.run(); expect(a.num).toEqual();
});
});

总结

在比较简单的情况下(如需要哑对象来通过编译,或是需要测试桩来替换耦合的组件),使用Stub。

如果需要验证耦合组件的行为,可以使用Spy或Mock。

参考资料

软件测试- 3 - Mock 和Stub的区别

浅谈mock和stub

《xUnit测试模式--测试码重构》

我对Stub和Mock的理解的更多相关文章

  1. Stub和Mock的理解

    我对Stub和Mock的理解 介绍 使用测试驱动开发大半年了,我还是对Stub和Mock的认识比较模糊,没有进行系统整理. 今天查阅了相关资料,觉得写得很不错,所以我试图在博文中对资料进行整理一下,再 ...

  2. [转]TDD之Dummy Stub Fake Mock

    TDD之Dummy Stub Fake Mock 测试驱动大家都很熟悉了,这两天正好看了一个java的书,对TDD中的一些基本概念进行了复习,具体如下: Dummy An object that is ...

  3. What's the difference between a stub and mock?

    I believe the biggest distinction is that a stub you have already written with predetermined behavio ...

  4. TDD 中关于mock一些理解

    最近在写代码的UT时case注意到: 在写某个类的test suit时,如果这个类既有组合(Composition),又有聚合关系(Aggregation). 组合关系(Composition):部分 ...

  5. 从0开发3D引擎(四):搭建测试环境

    目录 上一篇博文 了解自动化测试 单元测试 集成测试 端对端测试 通过打印日志来调试 了解运行测试 断点调试 通过Spector.js测试WebGL 通过log调试Shader 移动端测试 了解性能测 ...

  6. [转]软件测试- 3 - Mock 和Stub的区别

    由于一直没有完全搞明白Mock和Stub的区别,所以查了很多文章,而这一篇是做好的: http://yuan.iteye.com/blog/470418 尤其是8楼,Frostred的发言,描述地相当 ...

  7. 置换测试: Mock, Stub 和其他

    简介 在理想情况下,你所做的所有测试都是能应对你实际代码的高级测试.例如,UI 测试将模拟实际的用户输入(Klaas 在他的文章中有讨论)等等.实但际上,这并非永远都是个好主意.为每个测试用例都访问一 ...

  8. TDD:什么是桩(stub)和模拟(mock)?

    背景 本文假设你对TDD.Stub和Mock已经有了初步的认识,本文简单解释一下stub和mock的区别和使用场景,先看看他们之间的关系: 由上图可以知道mock框架可以非常容易的开发stub和moc ...

  9. TDD学习笔记【六】一Unit Test - Stub, Mock, Fake 简介

    这篇文章简介一下,如何通过 mock framework,来辅助我们更便利地模拟目标对象的依赖对象,而不必手工敲堆只为了这次测试而存在的辅助类型. 而模拟目标对象的部分,常见的有 stub objec ...

随机推荐

  1. 多线程CountDownLatch和Join

    如果现在有五个线程A.B.C.D.E,请问如何用E线程用于统计A.B.C.D四个线程的结果? 题意需要用E线程统计A.B.C.D四个线程,也就是说E线程必须要等到前面四个线程运行结束之后才能执行.那么 ...

  2. 更换gitlab公网IP,引发的故障。

    gitlab更换公网IP地址,导致gitlab非常的很卡,并且ssh方式添加的远程仓库是无法git pull 或者git push,只有是添加http方式的可以正常git pull和git push ...

  3. ecshop点滴记录

    会员中心: 用户中心页面的内容分布在两个模板文件中: user_clips.dwt(包含:欢迎页.我的留言.我的评论.我的标签.收藏商品.缺货登记.添加缺货登 记.我的推荐.单个商品推荐) user_ ...

  4. 老李分享:接电话之uiautomator 1

    老李分享:接电话之uiautomator   poptest是国内唯一一家培养测试开发工程师的培训机构,以学员能胜任自动化测试,性能测试,测试工具开发等工作为目标.如果对课程感兴趣,请大家咨询qq:9 ...

  5. Linux命令速查大全

    常用基本命令 ls 显示文件或目录 -l 列出文件详细信息l(list) -a 列出当前目录下所有文件及目录,包括隐藏的a(all) mkdir 创建目录 -p 创建目录,若无父目录,则创建p(par ...

  6. ASP.NET Core 网站发布到Linux服务器

    长期以来,使用.NET开发的应用只能运行在Windows平台上面,而目前国内蓬勃发展的互联网公司由于成本的考虑,大量使用免费的Linux平台,这就使得.NET空有一身绝技但无法得到广大的施展空间,.N ...

  7. [转]html中offsetTop、clientTop、scrollTop、offsetTop各属性介绍

    HTML精确定位:scrollLeft,scrollWidth,clientWidth,offsetWidth scrollHeight: 获取对象的滚动高度. scrollLeft:设置或获取位于对 ...

  8. 爬虫入门系列(二):优雅的HTTP库requests

    在系列文章的第一篇中介绍了 HTTP 协议,Python 提供了很多模块来基于 HTTP 协议的网络编程,urllib.urllib2.urllib3.httplib.httplib2,都是和 HTT ...

  9. AFNetworking 内部详解

    AFNetworking 是一个适用于IOS 和 Mac OSX 两个平台的网络库,他是在Foundation URL Loading System  基础上进行的一套封装 ,并提供了丰富的API接口 ...

  10. iOS 播放GIf图, 动态效果

    一.如果你集成了SDWebImage , 有一个很简单的方法 //导入sdwebImage的某个头文件 #import "UIImage+GIF.h" _bubble1.backg ...