Mock 框架 Moq 的使用

Intro

Moq 是 .NET 中一个很流行的 Mock 框架,使用 Mock 框架我们可以只针对我们关注的代码进行测试,对于依赖项使用 Mock 对象配置预期的依赖服务的行为。

Moq 是基于 Castle 的动态代理来实现的,基于动态代理技术动态生成满足指定行为的类型

在一个项目里, 我们经常需要把某一部分程序独立出来以便我们可以对这部分进行测试. 这就要求我们不要考虑项目其余部分的复杂性, 我们只想关注需要被测试的那部分. 这里就需要用到模拟(Mock)技术.

因为, 请仔细看. 我们想要隔离测试的这部分代码对外部有一个或者多个依赖. 所以编写测试代码的时候, 我们需要提供这些依赖. 而针对隔离测试, 并不应该使用生产时用的依赖项, 所以我们使用模拟版本的依赖项, 这些模拟版依赖项只能用于测试时, 它们会使隔离更加容易.

img

绿色的是需要被测试的类,黄色Mock的依赖项

——引用自杨旭大佬的博文

Prepare

首先我们需要先准备一下用于测试的类和接口,下面的示例都是基于下面定义的类和方法来做的

public interface IUserIdProvider
{string GetUserId();
}
public class TestModel
{public int Id { get; set; }
}
public interface IRepository
{int Version { get; set; }int GetCount();Task<int> GetCountAsync();TestModel GetById(int id);List<TestModel> GetList();TResult GetResult<TResult>(string sql);int GetNum<T>();bool Delete(int id);
}public class TestService
{private readonly IRepository _repository;public TestService(IRepository repository){_repository = repository;}public int Version{get => _repository.Version;set => _repository.Version = value;}public List<TestModel> GetList() => _repository.GetList();public TResult GetResult<TResult>(string sql) => _repository.GetResult<TResult>(sql);public int GetResult(string sql) => _repository.GetResult<int>(sql);public int GetNum<T>() => _repository.GetNum<T>();public int GetCount() => _repository.GetCount();public Task<int> GetCountAsync() => _repository.GetCountAsync();public TestModel GetById(int id) => _repository.GetById(id);public bool Delete(TestModel model) => _repository.Delete(model.Id);
}

我们要测试的类型就是类似 TestService 这样的,而 IRepositoy<TestModel>IUserIdProvider 是属于外部依赖

Mock Method

Get Started

通常我们使用 Moq 最常用的可能就是 Mock 一个方法了,最简单的一个示例如下:

[Fact]
public void BasicTest()
{var userIdProviderMock = new Mock<IUserIdProvider>();userIdProviderMock.Setup(x => x.GetUserId()).Returns("mock");Assert.Equal("mock", userIdProviderMock.Object.GetUserId());
}

Match Arguments

通常我们的方法很多是带有参数的,在使用 Moq 的时候我们可以通过设置参数匹配为不同的参数返回不同的结果,来看下面的这个例子:

[Fact]
public void MethodParameterMatch()
{var repositoryMock = new Mock<IRepository>();repositoryMock.Setup(x => x.Delete(It.IsAny<int>())).Returns(true);repositoryMock.Setup(x => x.GetById(It.Is<int>(_ => _ > 0))).Returns((int id) => new TestModel(){Id = id});var service = new TestService(repositoryMock.Object);var deleted = service.Delete(new TestModel());Assert.True(deleted);var result = service.GetById(1);Assert.NotNull(result);Assert.Equal(1, result.Id);result = service.GetById(-1);Assert.Null(result);repositoryMock.Setup(x => x.GetById(It.Is<int>(_ => _ <= 0))).Returns(() => new TestModel(){Id = -1});result = service.GetById(0);Assert.NotNull(result);Assert.Equal(-1, result.Id);
}

通过 It.IsAny<T> 来表示匹配这个类型的所有值,通过 It.Is<T>(Expression<Func<bool>>) 来设置一个表达式来断言这个类型的值

通过上面的例子,我们可以看的出来,设置返回值的时候,可以直接设置一个固定的返回值,也可以设置一个委托来返回一个值,也可以根据方法的参数来动态配置返回结果

Async Method

现在很多地方都是在用异步方法,Moq 设置异步方法有三种方式,一起来看一下示例:

[Fact]
public async Task AsyncMethod()
{var repositoryMock = new Mock<IRepository>();// Task.FromResultrepositoryMock.Setup(x => x.GetCountAsync()).Returns(Task.FromResult(10));// ReturnAsyncrepositoryMock.Setup(x => x.GetCountAsync()).ReturnsAsync(10);// Mock Result, start from 4.16repositoryMock.Setup(x => x.GetCountAsync().Result).Returns(10);var service = new TestService(repositoryMock.Object);var result = await service.GetCountAsync();Assert.True(result > 0);
}

还有一个方式也可以,但是不推荐,编译器也会给出一个警告,就是下面这样

repositoryMock.Setup(x => x.GetCountAsync()).Returns(async () => 10);

Generic Type

有些方法会是泛型方法,对于泛型方法,我们来看下面的示例:

[Fact]
public void GenericType()
{var repositoryMock = new Mock<IRepository>();var service = new TestService(repositoryMock.Object);repositoryMock.Setup(x => x.GetResult<int>(It.IsAny<string>())).Returns(1);Assert.Equal(1, service.GetResult(""));repositoryMock.Setup(x => x.GetResult<string>(It.IsAny<string>())).Returns("test");Assert.Equal("test", service.GetResult<string>(""));
}[Fact]
public void GenericTypeMatch()
{var repositoryMock = new Mock<IRepository>();var service = new TestService(repositoryMock.Object);repositoryMock.Setup(m => m.GetNum<It.IsAnyType>()).Returns(-1);repositoryMock.Setup(m => m.GetNum<It.IsSubtype<TestModel>>()).Returns(0);repositoryMock.Setup(m => m.GetNum<string>()).Returns(1);repositoryMock.Setup(m => m.GetNum<int>()).Returns(2);Assert.Equal(0, service.GetNum<TestModel>());Assert.Equal(1, service.GetNum<string>());Assert.Equal(2, service.GetNum<int>());Assert.Equal(-1, service.GetNum<byte>());
}

如果要 Mock 指定类型的数据,可以直接指定泛型类型,如上面的第一个测试用例,如果要不同类型设置不同的结果一种是直接设置类型,如果要指定某个类型或者某个类型的子类,可以用 It.IsSubtype<T>,如果要指定值类型可以用 It.IsValueType,如果要匹配所有类型则可以用 It.IsAnyType

Callback

我们在设置 Mock 行为的时候可以设置 callback 来模拟方法执行时的逻辑,来看一下下面的示例:

[Fact]
public void Callback()
{var deletedIds = new List<int>();var repositoryMock = new Mock<IRepository>();var service = new TestService(repositoryMock.Object);repositoryMock.Setup(x => x.Delete(It.IsAny<int>())).Callback((int id) =>{deletedIds.Add(id);}).Returns(true);for (var i = 0; i < 10; i++){service.Delete(new TestModel() { Id = i });}Assert.Equal(10, deletedIds.Count);for (var i = 0; i < 10; i++){Assert.Equal(i, deletedIds[i]);}
}

Verification

有时候我们会验证某个方法是否执行,并不需要关注是否方法的返回值,这时我们可以使用 Verification 验证某个方法是否被调用,示例如下:

[Fact]
public void Verification()
{var repositoryMock = new Mock<IRepository>();var service = new TestService(repositoryMock.Object);service.Delete(new TestModel(){Id = 1});repositoryMock.Verify(x => x.Delete(1));repositoryMock.Verify(x => x.Version, Times.Never());Assert.Throws<MockException>(() => repositoryMock.Verify(x => x.Delete(2)));
}

如果方法没有被调用,就会引发一个 MockException 异常:

verification failed

Verification 也可以指定方法触发的次数,比如:repositoryMock.Verify(x => x.Version, Times.Never);,默认是 Times.AtLeastOnce,可以指定具体次数 Times.Exactly(1) 或者指定一个范围 Times.Between(1,2, Range.Inclusive),Moq 也提供了一些比较方便的方法,比如Times.Never()/Times.Once()/Times.AtLeaseOnce()/Times.AtMostOnce()/Times.AtLease(2)/Times.AtMost(2)

Mock Property

Moq 也可以 mock 属性,property 的本质是方法加一个字段,所以也可以用 Mock 方法的方式来 Mock 属性,只是使用 Mock 方法的方式进行 Mock 属性的话,后续修改属性值就不会引起属性值的变化了,如果修改属性,则要使用 SetupProperty 的方式来 Mock 属性,具体可以参考下面的这个示例:

[Fact]
public void Property()
{var repositoryMock = new Mock<IRepository>();var service = new TestService(repositoryMock.Object);repositoryMock.Setup(x => x.Version).Returns(1);Assert.Equal(1, service.Version);service.Version = 2;Assert.Equal(1, service.Version);
}[Fact]
public void PropertyTracking()
{var repositoryMock = new Mock<IRepository>();var service = new TestService(repositoryMock.Object);repositoryMock.SetupProperty(x => x.Version, 1);Assert.Equal(1, service.Version);service.Version = 2;Assert.Equal(2, service.Version);
}

Sequence

我们可以通过 Sequence 来指定一个方法执行多次返回不同结果的效果,看一下示例就明白了:

[Fact]
public void Sequence()
{var repositoryMock = new Mock<IRepository>();var service = new TestService(repositoryMock.Object);repositoryMock.SetupSequence(x => x.GetCount()).Returns(1).Returns(2).Returns(3).Throws(new InvalidOperationException());Assert.Equal(1, service.GetCount());Assert.Equal(2, service.GetCount());Assert.Equal(3, service.GetCount());Assert.Throws<InvalidOperationException>(() => service.GetCount());
}

第一次调用返回值是1,第二次是2,第三次是3,第四次是抛了一个 InvalidOperationException

LINQ to Mocks

我们可以通过 Mock.Of 来实现类似 LINQ 的方式,创建一个 mock 对象实例,指定类型的实例,如果对象比较深,要 mock 的对象比较多使用这种方式可能会一定程度上简化自己的代码,来看使用示例:

[Fact]
public void MockLinq()
{var services = Mock.Of<IServiceProvider>(sp =>sp.GetService(typeof(IRepository)) == Mock.Of<IRepository>(r => r.Version == 1) &&sp.GetService(typeof(IUserIdProvider)) == Mock.Of<IUserIdProvider>(a => a.GetUserId() == "test"));Assert.Equal(1, services.ResolveService<IRepository>().Version);Assert.Equal("test", services.ResolveService<IUserIdProvider>().GetUserId());
}

Mock Behavior

默认的 Mock Behavior 是 Loose,默认没有设置预期行为的时候不会抛异常,会返回方法返回值类型的默认值或者空数组或者空枚举,

在声明 Mock 对象的时候可以指定 Behavior 为 Strict,这样就是一个**"真正"**的 mock 对象,没有设置预期行为的时候就会抛出异常,示例如下:

[Fact]
public void MockBehaviorTest()
{// Make mock behave like a "true Mock",// raising exceptions for anything that doesn't have a corresponding expectation: in Moq slang a "Strict" mock;// default behavior is "Loose" mock,// which never throws and returns default values or empty arrays, enumerable, etcvar repositoryMock = new Mock<IRepository>();var service = new TestService(repositoryMock.Object);Assert.Equal(0, service.GetCount());Assert.Null(service.GetList());var arrayResult = repositoryMock.Object.GetArray();Assert.NotNull(arrayResult);Assert.Empty(arrayResult);repositoryMock = new Mock<IRepository>(MockBehavior.Strict);Assert.Throws<MockException>(() => new TestService(repositoryMock.Object).GetCount());
}

使用 Strict 模式不设置预期行为的时候就会报异常,异常信息类似下面这样:

strict exception

More

Moq 还有一些别的用法,还支持事件的操作,还有 Protected 成员的 Mock,还有一些高级的用法,自定义 Default 行为等,感觉我们平时可能并不太常用,所以上面并没有加以介绍,有需要用的可以参考 Moq 的文档

上述测试代码可以在 Github 获取 https://github.com/WeihanLi/SamplesInPractice/blob/master/XunitSample/MoqTest.cs

References

  • https://github.com/moq/moq4/wiki/Quickstart

  • https://github.com/moq/moq4

  • https://github.com/WeihanLi/SamplesInPractice/blob/master/XunitSample/MoqTest.cs

  • https://www.cnblogs.com/tylerzhou/p/11410337.html

  • https://www.cnblogs.com/cgzl/p/9304567.html

  • https://www.cnblogs.com/haogj/archive/2011/07/22/2113496.html

Mock 框架 Moq 的使用相关推荐

  1. 【C#】【xUnit】【Moq】.NET单元测试Mock框架Moq初探!

    在TDD开发模型中,经常是在编码的同时进行单元测试的编写,由于现代软件开发不可能是一个人完成的工作,所以在定义好接口的时候我们就可以进行自己功能的开发(接口不能经常变更),而我们调用他人的功能时只需要 ...

  2. C#/.net 单元测试xUnit、Mock、Moq

    C#/.net 单元测试xUnit.Mock.Moq 在做单元测试的时候,有时需要引用很多的外部对象,例如网络通信.记录日志等.单元测试无法控制这些外部的依赖对象,所以需要使用Stub和Mock来模拟 ...

  3. Mock和Java单元测试中的Mock框架Mockito介绍

    什么是Mock? 在面向对象程序设计中,模拟对象(英语:mock object,也译作模仿对象)是以可控的方式模拟真实对象行为的假的对象.程序员通常创造模拟对象(mock object)来测试其他对象 ...

  4. 单元测试mock框架——jmockit实战

    JMockit是google code上面的一个java单元测试mock项目,她很方便地让你对单元测试中的final类,静态方法,构造方法进行mock,功能强大.项目地址在:http://jmocki ...

  5. Java单元测试之Mock框架

    一.引言 二.为什么要用Mock 三.Mock使用场景 四.Mock定义 五.Mock框架 五.Mockito 5.1 Mockito基本使用 5.2 MockMVC测试 5.2.1 初始化MockM ...

  6. Mock框架的三次迭代,让你的单元测试更高效

    如何定义单元 对于单元测试中的单元,不同的人有不同的看法:可以理解为一个方法,可以理解为一个完整的接口实现,也可以理解为一个完整的功能模块或者是多个功能模块的一个耦合. 根据以往的单元测试经验,在设计 ...

  7. MoQ(基于.net3.5,c#3.0的mock框架)简单介绍(转)

    https://www.cnblogs.com/nuaalfm/archive/2009/11/25/1610755.html 转载于:https://www.cnblogs.com/lip-blog ...

  8. Mock框架Mockito入门教程

    在开发中,我们经常会依赖同事或者第三方提供的接口,如果该接口无法正常工作:比如接口正在修复,或者网络异常等.那么便会对需要依赖该接口的开发造成很大影响. 遇到这种情况,我们可能会想到模拟该接口以提供正 ...

  9. 单元测试Mock框架--Mockito

    文章目录 目前开发中,单元测试遇到的问题 解决方案--Mock Junit4 + Mockito: Mockito常用注解: Mockito常用方法: Tips: 总结 目前开发中,单元测试遇到的问题 ...

最新文章

  1. 下列哪个不是python合法的变量名_下列哪个不是Python中合法的数据类型?
  2. python中的引用、浅拷贝和深拷贝
  3. sdwan架构怎么搭建?
  4. LeetCode 996. 正方形数组的数目(回溯+剪枝)
  5. [转载] Python repr() 函数
  6. [转载] pandas中Series数组创建方法
  7. python编程的区别_Python与其它编程语言的区别
  8. 台式计算机常用总线,计算机中常见的总线有哪些
  9. Air780E模块PPP应用开发指南
  10. Push上传出错:Support for password authentication was removed on August 13, 2021.git did not exit cleanly
  11. unity VR中制作小地图的方法
  12. XXL-Job Docker部署
  13. 两个鸡蛋和一百层楼的问题
  14. docker(4): 持久化
  15. 12 More Effective C++—条款16/17 (2/8原理与延缓求值)
  16. 容器云职业技能大赛 不一样的比赛
  17. 龙门阵179期实录:技术专场之Android安全现状
  18. 高阻态是0还是1_超实用买鞋指南!宽脚、胖脚、脚背高、磨脚...搞定各种买鞋难题!...
  19. 作为一名IT狗,天天加班,快变秃子了,我决定去植发……
  20. remapkey不能打开注册表_Windows技巧之妙用注册表

热门文章

  1. c++万能头文件_初学Python,与C对比
  2. 使用HTML5、CSS3和jQuery增强网站用户体验
  3. MSSqlServer基础学习01
  4. 小米oj 反向位整数(简单位运算)
  5. JDBC 学习笔记(一)—— JDBC 基础
  6. 求指教、。。。关于调用so文件
  7. MySQL5.5多实例编译安装——mysqld_multi
  8. poj1189 简单dp
  9. NSInteger,NSUInteger,NSNumber
  10. (转)前置++和后置++的区别