介绍

使用测试驱动开发大半年了,我还是对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. git 入门宝典

    本篇教程是按照我自己的组织方式,然后从多篇教程中拼凑出来的,嘎嘎~,真佩服自己的技术! 原本想叫 git 宝典的,结果一查git的命令大全,还有那么多的git命令与功能没有接触到,所以...还是谦虚一 ...

  2. laravel 框架memcache的配置

    Laravel5框架在Cache和Session中不支持Memcache,看清了是Memcache而不是Memcached哦,MemCached是支持的但是这个扩展真的是装的蛋疼,只有修改部分源码让其 ...

  3. python 自动化接口测试(6)

    迎接新的一波更新吧,这次是基于图灵机器人的一个api接口的测试. 这是api的接口:http://www.tuling123.com/openapi/api 我们试着通过浏览器直接访问看下 这是反馈的 ...

  4. ajax第三步

    ajaxSend()函数用于设置当AJAX请求即将被发送时执行的回调函数. 这是一个全局AJAX事件函数,用于为任何AJAX请求的ajaxSend事件绑定事件处理函数.当AJAX请求即将被发送时,将触 ...

  5. ASP.NET自定义处理程序

    要创建自定义处理程序,可以创建一个实现IHttpHandler接口的类. 该类有两个重要的参数:IsResuable属性和ProcessRequest方法.如果处理程序实例可以在不同的请求中重用,Is ...

  6. 深入Callable及Runnable两个接口 获取线程返回结果

    今天碰到一个需要获取线程返回结果的业务场景,所以了解到了Callable接口. 先来看下下面这个例子: public class ThreadTest { public static void mai ...

  7. Unity 动态加载 Prefab

    Unity3D 里有两种动态加载机制:一个是Resources.Load,另外一个通过AssetBundle,其实两者区别不大. Resources.Load就是从一个缺省打进程序包里的AssetBu ...

  8. 【STL】reverse函数用法

    reverse函数的功能是反转排序一个容器中指定元素的内容. 函数参数:reverse(first,last), 其中first,last分别指向被反转序列中初始及末尾位置的双向迭代器(Bidirec ...

  9. 一个全局变量引起的DLL崩溃

    参考我发的帖子: http://bbs.csdn.net/topics/390737064?page=1#post-397000946 现象是exe程序在加载dll的时候崩溃了,莫名其妙的崩溃了.换其 ...

  10. Android 性能优化——之控件的优化

    Android 性能优化——之控件的优化 前面讲了图像的优化,接下来分享一下控件的性能优化,这里主要是面向自定义View的优化. 1.首先先说一下我们在自定义View中可能会犯的3个错误: 1)Use ...