Mock 框架 Moq 的使用
Mock 框架 Moq 的使用
Intro
Moq 是 .NET 中一个很流行的 Mock 框架,使用 Mock 框架我们可以只针对我们关注的代码进行测试,对于依赖项使用 Mock 对象配置预期的依赖服务的行为。
Moq 是基于 Castle
的动态代理来实现的,基于动态代理技术动态生成满足指定行为的类型
在一个项目里, 我们经常需要把某一部分程序独立出来以便我们可以对这部分进行测试. 这就要求我们不要考虑项目其余部分的复杂性, 我们只想关注需要被测试的那部分. 这里就需要用到模拟(Mock)技术.
因为, 请仔细看. 我们想要隔离测试的这部分代码对外部有一个或者多个依赖. 所以编写测试代码的时候, 我们需要提供这些依赖. 而针对隔离测试, 并不应该使用生产时用的依赖项, 所以我们使用模拟版本的依赖项, 这些模拟版依赖项只能用于测试时, 它们会使隔离更加容易.
绿色的是需要被测试的类,黄色是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.FromResult
repositoryMock.Setup(x => x.GetCountAsync())
.Returns(Task.FromResult(10));
// ReturnAsync
repositoryMock.Setup(x => x.GetCountAsync())
.ReturnsAsync(10);
// Mock Result, start from 4.16
repositoryMock.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
也可以指定方法触发的次数,比如: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, etc
var 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
模式不设置预期行为的时候就会报异常,异常信息类似下面这样:
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 的使用的更多相关文章
- .NET Core之单元测试(三):Mock框架Moq的使用
编写一个API 新增一个接口 public interface IFoo { bool Ping(string ip); } 接口实现 public class Foo : IFoo { public ...
- MoQ(基于.net3.5,c#3.0的mock框架)简单介绍
我们在做单元测试的时候,常常困扰于数据的持久化问题,很多情况下我们不希望单元测试影响到数据库中的内容,而且受数据库的影响有时我们的单元测试的速度会很慢,所以我们往往希望将持久化部分隔离开,做单元测试的 ...
- Mock框架
MoQ(基于.net3.5,c#3.0的mock框架)简单介绍 - 你听海是不是在笑 - 博客园 http://www.cnblogs.com/nuaalfm/archive/2009/11/25/1 ...
- 教育单元测试mock框架优化之路(中)
转载:https://sq.163yun.com/blog/article/169564470918451200 三.间接依赖的bean的mock替换 对于前面提供的@Mock,@Spy+@Injec ...
- 教育单元测试mock框架优化之路(下)
转载:https://sq.163yun.com/blog/article/169563599967031296 四.循环依赖的解决 果然! 当我将@SpyBean应用到存在有循环依赖的Bean上时, ...
- 教育单元测试mock框架优化之路(上)
转载:https://sq.163yun.com/blog/article/169561874192850944 众所周知,mock对于单元测试,尤其是基于spring容器的单元测试,是非常重要的.它 ...
- 单元测试mock框架——jmockit实战
JMockit是google code上面的一个java单元测试mock项目,她很方便地让你对单元测试中的final类,静态方法,构造方法进行mock,功能强大.项目地址在:http://jmocki ...
- 推荐一款数据mock框架,无需任何依赖,贼牛逼
fox-mock 是基于Java Agent实现的自测,联调Mock利器.能解决你的这些问题: 开发过程中,依赖了下游多个接口,想跑个单测都必须得等下游把服务部署好 联调过程中,下游某个接口出问题,阻 ...
- MoQ(基于.net3.5,c#3.0的mock框架)简单介绍(转)
https://www.cnblogs.com/nuaalfm/archive/2009/11/25/1610755.html
随机推荐
- JavaScript——内置对象
- MHA 的 Binlog Server & VIP 漂移
目录 Binlog Server 在 MHA 配置文件中配置 Binlog Server 创建 Binlog 存放目录 实时传输主库 Binlog 命令 重启 MHA 检验 MHA Manager 服 ...
- Linux-用户/用户组身份提权
sudo 身份提权(更安全) su命令在切换用户身份时,如果每个普通用户都能拿到root用户的密码,当其中某个用户不小心泄漏了root的密码,那系统会变得非常不安全. 为了改进这个问题,从而产生了su ...
- 1076D Edge Deletion 【最短路】
题目:戳这里 题意:求出1到所有点的最短路径后,把边减到小于等于k条,问保留哪些边可以使仍存在的最短路径最多. 解题思路:这题就是考求最短路的原理.比如dijkstra,用优先队列优化后存在队列中的前 ...
- C# 类 (6) -继承
继承 定义类的时候,public class Dog:Animal 表示 Dog 这个类是 继承自 Animal,冒号后面的是它的基类 继承后 的Dog 类,当调用Dog.Great() 的时候输出的 ...
- 操作系统 part3
1.操作系统四特性 并发:一个时间段,多个进程在宏观上同时运行 共享:系统中的资源可以被多个并发进程共同使用(互斥共享,同时共享) 虚拟:利用多道程序设计,利用时分复用(分时系统)和空分复用(虚拟内存 ...
- 牛客多校第九场H Cutting Bamboos(主席树 区间比k小的个数)题解
题意: 标记为\(1-n\)的竹子,\(q\)个询问,每次给出\(l,r,x,y\).要求为砍区间\(l,r\)的柱子,要求砍\(y\)次把所有竹子砍完,每次砍的时候选一个高度,把比他高的都砍下来,并 ...
- 浅谈WEB前端规范化标准之ESlint
规范化标准 软件开发需要多人开发,不同的开发者具有不同的编码习惯和喜好,不同的喜好增加项目的维护成本,所以需要明确统一的标准,决定 了项目的可维护性,人为的约定不可靠,所以需要专门的工具进行约束,并且 ...
- 编程方式建视频——GitHub 热点速览 v.21.07
作者:HelloGitHub-小鱼干 假期过半,大家过得如何,吃好喝好了吗?GitHub 很好!本周的 GitHub Trending 又上爆款项目--github1s 装完之后,一秒 GitHub ...
- Vue & Sentry & ErrorHandler
Vue & Sentry & ErrorHandler import * as Sentry from '@sentry/browser'; import { Vue as VueIn ...