我对Stub和Mock的理解

介绍

使用测试驱动开发大半年了,我还是对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 = 0;
}
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 1;
}
} var a = new A();
a.run(); expect(a.num).toEqual(1);
});
});

示例2(替换类):

//产品代码
function A() {
this.num = 0;
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 1;
}; var a = new A();
a.run(); expect(a.num).toEqual(1);
});
});

Test Stub(测试桩)

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

示例:

//产品代码
function A(b) {
this.num = 0;
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 1;
}
}; var a = new A(stub_B); //注入桩
a.run(); expect(a.num).toEqual(1);
});
});

Test Spy(嗅探桩)

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

示例:

可以使用jasmine的spy来举例。

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

Mock Object(模拟对象)

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

示例:

//产品代码
function A(b) {
this.num = 0;
this._b = b;
}
A.prototype.run = function () {
this.num = this._b.getNum(2);
}; //测试代码
describe("测试A类的run方法", function () {
it("获得数字", function () {
var mockB = Mock.createMock({
getNum: function(){}
}); //如果B类存在的话,也可以直接传入B的原型:var mockB = Mock.createMock(B.prototype);
Mock.expect(mockB.getNum, 2).return(1).times(1); var a = new A(mockB);
a.run(); expect(a.num).toEqual(1);
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 = 0;
this._b = b;
}
A.prototype.run = function () {
this.num = this._b.getNum(2);
}; //测试代码
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(1); //只指定返回值,没有期望的参数或期望调用的次数。因此不用verify来验证了! var a = new A(mockB);
a.run(); expect(a.num).toEqual(1);
});
});

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

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

总结

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

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

参考资料

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

浅谈mock和stub

30天快速上手 TDD Day 7 - Unit Test - Stub, Mock, Fake 簡介

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

 

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

  1. 我对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. 经验28--相关时间戳,C#

    时间戳通常用于设置独特性质,保存图片之类的,到文件名后添加. 时间戳一般17地点. 1.获取的当前时间的时间戳. DateTime dtStart = TimeZone.CurrentTimeZone ...

  2. 宽客的人&&事件映射

    看完<宽客>这本书,叙事介绍20世纪华尔街对冲基金.股票.投资者依赖股市从直觉交易数学家的早期演化.物理学家用数学模型开发过程中的交易,这些进入金融数学家.物理学家依靠大数据分析.稍纵即逝 ...

  3. Android 自己的自动化测试(5)&lt;robotium&gt;

    大约Android自己的自动化测试UI测试,前出台Android 自己主动化測试(4)<uiautomator>, 在android原生的单元測试框架上,利用uiautomator.jar ...

  4. 教你一步一步部署.net免费空间OpenShift系列之二------创建应用

    接上回书,注册完毕后需要在Openshift上创建一个应用空间,如如何创建空间呢,Openshift本身是不直接支持Mono来部署ASP.Net程序的,需要借助openshift-community- ...

  5. UVA - 11986 Save from Radiation

    Description J Save from Radiation Most of you are aware of Nuclear Power Plant Explosion at Fukushim ...

  6. Atitit.软件GUIbutton与仪表盘--db数据库区--导入mysql sql错误的解决之道

    Atitit.软件GUIbutton与仪表盘--db数据库区--导入mysql sql错误的解决之道 Keyword::截取文本文件后部分 查看提示max_allowed_packet限制 Targe ...

  7. Coreseek/sphinx全文检索的了解

    Coreseek/sphinx全文检索的了解 概述: 全文检索是一种将文件里全部文本与检索项匹配的文字资料检索方法,全文检索是将存储于数据库中整本书.整篇文章中的随意内容信息查找出来的检索.它能够依据 ...

  8. jQuery选中该复选框来实现/全部取消/未选定/获得的选定值

    <!DOCTYPE html> <html> <head> <script type="text/javascript" src=&quo ...

  9. Installshield停止操作系统进程的代码--IS5版本适用

    原文:Installshield停止操作系统进程的代码--IS5版本适用 出处:http://www.installsite.org/pages/en/isp_ext.htm这个地址上有不少好东西,有 ...

  10. [SQL]死锁处理语句

    原文:[SQL]死锁处理语句 引言 今天在群里看到分享的解决死锁的sql语句,就想着这东西以后肯定用的着,就下载下来,在这里记录一下,以后查找也方便. SQL SET QUOTED_IDENTIFIE ...