一文让你掌握单元测试的Mock、Stub和Fake

单元测试中有几个神秘的概念,它们就是Mock,模拟对象;Stub,存根;Fake,伪对象,它们听起来很类似,也很容易混淆,让我们通过这篇文章揭开它们神秘的面纱,探索其幽深的小径。

1.什么是伪对象(Fake)

伪对象,通俗的将就是假货


是用来代替具有“智能”对象的假货实现。通常是一个快捷实现,使它在不同的单元测试中有用,但不能用作集成测试。

到目前为止,我看到的最常见的例子是数据仓储层中。假设我有一个标准的 SQL Server 仓储库,如下所示:

public interface IUserRepository
{void Insert(object user);List<object> GetAllUsers();
}public class UserRepository : IUserRepository
{public List<object> GetAllUsers(){//到数据库取用户集. }public void Insert(object user){//插入用户到数据库}
}

涉及到实际的实现部分时,可能包含逻辑和调用数据库的方法。

当涉及到对可能使用 IUserRepository 的类(例如 UserService)进行单元测试时,我们会遇到一些问题。因为我们不希望我们的单元测试接触到数据库,坦率地说,我们并不真正关心 UserRepository 的实现。

所以我们创建了一个伪对象,而不是直接使用已经实现的真实对象:

public class FakeUserRepository : IUserRepository
{private List<object> _users = new List<object>();public List<object> GetAllUsers(){return _users;}public void Insert(object user){_users.Add(user);}
}

在我们的 fake 中,我们实际上获取了插入的用户,并将其添加到内部列表中。当调用 GetAllUsers 时,我们返回相同的列表。现在,每当单元测试需要调用 IUserRepository 时,我们可以在 FakeUserRepository中进行补充,并且立即“工作”。

这里的主要内容是实现了业务上的相似!

这是一个“真正的”实现,实际上就像一个存储库一样,只是在幕后没有实际的数据库。

2.什么是存根(Stub)

存根是一种返回硬编码响应的实现.

存根没有任何“智能”。没有将对象上的调用捆绑在一起,而是每个方法只返回一个预定义的固定响应。

让我们看看如何为上述创建存根:

public class StubOneUserRepository : IUserRepository
{public List<object> GetAllUsers(){return new List<object>();}public void Insert(object user){//啥都不做~}
}

看起来它有点类似于我们的伪对象,但……不完全是。

这里插入不影响 GetAllUsers,GetAllUsers 本身返回一个没有任何内容的预设响应。我在测试期间对这个对象所做的任何事情都不会改变它的功能。

存根用于满足代码内部的条件,而不是测试功能。

如果我的代码在存储库上调用“插入”,但我并不真正关心我的特定测试的数据会发生什么,那么存根是有意义的,这就省去了编写伪对象“智能”业务的工作。

仓储库的例子显得有些奇葩,因为仓储库总是应该返回动态数据来测试代码中的各种条件。因此,让我使用另一个在现实世界中更有可能需要存根的示例。

假设有一个界面告诉用户是否经过“身份验证”。它看起来像这样:

public interface IUserAuthenticatedCheck
{bool IsUserAuthenticated();
}

现在对于我们的测试,总是需要对用户进行身份验证,也许是为了满足一些基础框架条件。可以像这样定义存根:

public class StubUserAuthenticatedCheckTrue : IUserAuthenticatedCheck
{//返回验证过~~~public bool IsUserAuthenticated() => true;
}

没有是否应该对用户进行身份验证的智能算法,没有其他值,只是一个直接的“总是返回真”的方法。

固定,就是存根擅长的地方。

3.什么是模拟对象(Mock)

模拟是一个预设的对象,可以将动态响应/行为定义为测试的一部分,并预先定义好。

它们不需要去特别实现或实例化,并且(通常)不需要在测试之间共享行为。

我们将在哪里使用 Mock 呢?是您想要相对动态的任何地方,对于特定测试满足条件。

假设我正在编写一个调用以下接口的测试:

public interface IShopService
{bool CheckShopIsOpen(int shopId);
}

我们所做的就是检查商店是开还是关。这个实际实现类可能会调用数据库或某种 webservice/api,但我们不想将其作为单元测试的一部分。

如果在这里使用Fake伪对象,我们需要添加一些虚拟方法来判断商店是应该开还是关。也许是这样的:

public class FakeShopService : IShopService
{public bool ShouldShopBeOpen { get; set; }public bool CheckShopIsOpen(int shopId){return ShouldShopBeOpen;}
}

呃,好像有些复杂,为了能够控制商店是开放还是关闭,我们需要添加新方法。

如果使用存根Stub,必须将真/假响应硬编码到具体类中。可能是这样的:

public class StubShopService : IShopService
{private Dictionary<int, bool> _shops = new Dictionary<int, bool>{{ 1, true },{ 2, false }};public bool CheckShopIsOpen(int shopId){return _shops[shopId];}
}

这适用于预定义的 id 列表,以及商店是开还是关。

但是如果您在测试中使用它并传入 1 的 id,从测试中并不能立即清楚为什么得到 true 的响应,可能需要回来看看你的硬编码?

那么如何使用模拟对象来解决这个问题?(当然建议你直接使用 Moq 库!):

var _mockShopService = new Mock<IShopService>();
_mockShopService.Setup(x => x.CheckShopIsOpen(1)).Returns(true);

就在测试代码中,当使用模拟对象 ID 为 1 的 CheckShopIsOpen 时,非常清楚,返回 true。

它也是特定于这个测试的,并且不会强迫我们在任何地方硬编码任何东西,或者创建具体的类。

当我们有一个测试要求商店 id 1 为假时…

_mockShopService.Setup(x => x.CheckShopIsOpen(1)).Returns(false);

so easy!

4. 何时使用 Mock、Fake 和 Stub

一切并不是绝对的,你懂得!

  • 当想要一个可重用的具体实现时,请使用 Fake,该实现与真实实现类似,具有跨测试的可重用性(例如内存数据库)

  • 当想要在测试中重复使用的硬编码响应/实现时使用存根Stub

  • 当需要对单个测试进行动态响应时使用 Mock

不过,一般测试可能并不会分的这么严格,大部分情况下,我只使用Mock!不服来战~~~~

好了,希望您能更好地理解这些测试对象的用途,并对何时使用每个对象有更多的了解。

最后,祝你好运!

5.小结

关注我,不迷路!看到这了,点个赞再走!
(思聪语录)别嫌我臭,我就是你的臭宝~~

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

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

相关文章

Jafka源码粗略解读之二--关于JMX

2019独角兽企业重金招聘Python工程师标准>>> JMX Jafka里用到了JMX&#xff0c;之前也没用过&#xff0c;迅速突击了一下&#xff0c;感觉还是挺简单的&#xff1a; 有一篇文章用一个例子介绍JMX怎么使用的&#xff0c;简洁明了&#xff1a;http://www.javalobby.or…

参数化的RBAC模型

1 动机 基于角色的访问控制(RBAC)模型被普遍认为是一种有效的访问控制模型&#xff0c;它比传统的自主访问控制(DAC)和强制访问控制(MAC)具有更高的灵活性和更好的扩展性。 在实际应用中&#xff0c;随着企业规模以及信息系统规模逐渐扩大&#xff0c;系统中角色的数目也随之急…

使用 ML.NET 进行保险价格预测

此前通过多篇文章已充分介绍过&#xff0c;ML.NET是一个开源的跨平台机器学习框架&#xff0c;特别适合 .NET 开发人员。它允许将机器学习集成到 .NET 应用中&#xff0c;而无需离开 .NET 生态系统&#xff0c;甚至拥有 ML 或数据科学背景。ML.NET 现有的各种内置模型训练器可用…

送礼物给女生,她哭了是怎么回事?

全世界只有3.14 % 的人关注了青少年数学之旅中秋节快要到了&#xff0c;超模君说要给我准备个惊喜&#xff0c;what&#xff1f;结果我在桌面上发现了一个盒子和一大堆 垃圾 零件&#xff0c;清洁阿姨你在哪&#xff1f;我需要你。不过仔细一看&#xff0c;我去&#xff1f;&am…

oracle+11g+rda,Oracle RDA 4.20 初体验

RDA 全名RemoteDiagnostic Agent&#xff0c;是Oracle用来收集、分析数据库的工具&#xff0c;但统计信息远远大于只是数据库的&#xff0c;也可以说是现在一个Oracle dba 角色需要掌握的Oracle DB SERVER的信息&#xff0c;包含数据库安装、配置、性能、备份等信息、操作系统各…

室内设计品牌网站搭建的作用是什么

随着人们生活质量日益提升&#xff0c;对其自身的居住环境也有了较高要求&#xff0c;每个人审美不一样&#xff0c;无论自己居住的房屋还是公司办公/商场等场景都需要设计不同的内容&#xff0c;还有各种设施的摆放及类别等都有讲究&#xff0c;尤其对公司及商场等环境&#x…

面向.NET开发人员的Dapr- actors 构建块

原文地址&#xff1a;https://docs.microsoft.com/en-us/dotnet/architecture/dapr-for-net-developers/actors The actor model originated in 1973. It was proposed by Carl Hewitt as a conceptual model of concurrent computation, a form of computing in which several…

史上最严重的忘拿钥匙事件 | 今日最佳

全世界只有3.14 % 的人关注了青少年数学之旅&#xff08;视频源网络&#xff0c;侵权删&#xff09;难搞噢↓ ↓ ↓

oracle经常开关好吗,频繁开关机对电脑有什么影响吗?

2005-08-03 08:01:45没关系的哦&#xff0c;一天五六次正常&#xff0c;只要是正常开关机就OK。全部2005-08-03 08:01:452005-08-03 07:49:392005-08-03 05:21:09影响不大完全在合理范围内,等到坏的时候也应该淘汰了全部2005-08-03 05:21:092005-08-03 04:09:04原则上经常性的开…

老是担心数学学不好?这些基础是时候正视了!

我们都知道&#xff0c;数学是学生生涯的一门重要学科&#xff0c;与其担心三年级掉队&#xff0c;不如从小培养良好的学习兴趣和数感直觉&#xff0c;之后的学习就是水到渠成的事了&#xff0c;这可不是报个奥数班就万事大吉了&#xff0c;最关键的&#xff0c;还得从培养孩子…

老师“鬼话”全曝光!哈哈哈哈哈哈全国的老师都这样吗?

老师您辛苦了明天就是教师节啦今天超模君有幸采访到一位从教多年的数学老师01请问您教的科目是&#xff1f;数学收到&#xff0c;over02您从教多久了&#xff1f;唔…快10年了您当过班主任吗&#xff1f;当过&#xff0c;现在也是班主任工作日常是怎样的&#xff1f;备课、改作…

单体系统如何拆分为微服务

当单体系统越来越大&#xff0c;并难于维护时&#xff0c;很多企业开始有意把单体系统拆分为微服务风格架构。这么做很有意义&#xff0c;但不容易。要做好这件事情我们必须学习&#xff0c;我们从一个简单的服务开始&#xff0c;另一方面拉出以垂直功能为基础的服务&#xff0…

Wiki及其他

大概是2年多以前&#xff0c;我几乎是和blog一起听到wiki的概念的。当时blog正备受推崇&#xff0c;而个性色彩稍逊一筹、讲究严肃协作的wiki则没怎么受公众注意。我也一样。后来进入行业之后&#xff0c;我一直想有一套知识库(Knowledge Base)系统&#xff0c;多分类、复杂查找…

linux服务器管理公司用户,在Linux服务器Jenkins中管理用户和角色的方法

下面将教你如何在Linux服务器Jenkins中管理用户和角色&#xff0c;它需要创建角色并分配给用户&#xff0c;你需要运行Jenkins服务器才能操作接下来的工作。安装Jenkins参考文章在Jenkins中管理用户和角色默认情况下&#xff0c;当你在Jenkins中创建用户时&#xff0c;它几乎可…

还在用Excel做数据分析?两大方法5分钟完成别人一天的工作

全世界只有3.14 % 的人关注了青少年数学之旅我是个只会用Excel的数据分析工作者。有一天&#xff0c;我和同事大鹏约好晚上一起喝酒&#xff0c;离下班还有5分钟&#xff0c;老板突然Q我&#xff1a;我怀着忐忑的心情打开了一个神秘的压缩包&#xff1a;912个CSV表格&#xff0…

poj1033

模拟题&#xff0c;注意不需要移动的情况要特殊输出 #include <cstdio> #include <cstring> #include <cstdlib> using namespace std;#define MAX_CLUSTER_NUM 10005int cluster_num, file_num; int link[MAX_CLUSTER_NUM]; bool is_free[MAX_CLUSTER_NUM];…

ABP Framework V4.4 RC 新增功能介绍

新增功能概述•启动模板删除 EntityFrameworkCore.DbMigrations 项目•CMS-Kit 模块新增 动态菜单管理 功能•对象扩展管理系统新增两个扩展方法&#xff1a;MapEfCoreDbContext MapEfCoreEntity&#xff0c;分别对 数据上下文和实体 自定义映射配置。•文本模板系统新增 Razor…

论文排版怕翻车?这个排版神器赶紧用起来!

论文的重要加分点除了内容&#xff0c;还有它相信每年临近准备毕设或者毕业论文的同学会遇到这些问题&#xff1a;“我的毕业论文提交一次就被导师批评一次&#xff0c;内容不行就算了&#xff0c;格式也有问题&#xff01;改论文改到绝望”“期刊的版式要求不是统一的&#xf…

[正则表达式] 可以解析HTML/XHTML页面的所有元素和结构的Regular Expression![ZT]

ZT: http://www.cnblogs.com/Laser_Lu/archive/2005/04/21/142605.html哈哈&#xff0c;继 昨天的那个正则表达式 之后又写了一个更长的Regular Expression&#xff0c;全长527&#xff0c;是用于查找出所有的XHTML/HTML的标记外面的所有空格&#xff0c;并将之转换为 的。 希望…

我用段子讲.NET之依赖注入(一)

我用段子讲.NET之依赖注入&#xff08;一&#xff09;1&#xff09;西城的某个人工湖畔&#xff0c;湖水清澈见底&#xff0c;湖畔柳树成荫。人工湖往北&#xff0c;坐落着两幢写字楼&#xff0c;水晶大厦靠近地铁站&#xff0c;由于为了与湖面天际线保持一致&#xff0c;楼层只…