使用 Moq 测试.NET Core 应用 -- Mock 属性

时间:2023-12-14 15:09:50

第一篇文章, 关于Mock的概念介绍: https://www.cnblogs.com/cgzl/p/9294431.html

第二篇文章, 关于方法Mock的介绍: https://www.cnblogs.com/cgzl/p/9300356.html

本文介绍Moq的使用.

使用的代码: https://github.com/solenovex/Moq4-Tutorial-Code 里面的 03 Before 部分.

Mock属性

属性是指 get set property.

接着上文, 我在03 Before部分的代码里做了一些修改.

首先IPhysicalExamination接口添加了IsMedicalRoomAvailable属性:

使用 Moq 测试.NET Core 应用 -- Mock 属性

其实现类:

使用 Moq 测试.NET Core 应用 -- Mock 属性

属性方法内依然没有做实现.

添加的这个属性在业务上的意思就是体检室是否可以使用. 如果不可以使用的话, 那么球员的转会操作应该被推迟.

所以还需要为转会结果枚举添加一个推迟:

使用 Moq 测试.NET Core 应用 -- Mock 属性

最后在转会审批逻辑里进行判断, 如果体检室不可用, 那么转会就被推迟:

使用 Moq 测试.NET Core 应用 -- Mock 属性

在单元测试里对属性进行mock非常的简单:

使用 Moq 测试.NET Core 应用 -- Mock 属性

这个测试也会通过的:

使用 Moq 测试.NET Core 应用 -- Mock 属性

递归Mock

修改一下IPhysicalExamination接口, 形成一个多层嵌套的属性:

使用 Moq 测试.NET Core 应用 -- Mock 属性

IPhysicalExamination --> IMedicalRoom --> IMedicalRoomStatus --> IsAvailable.

通过上面这一串来判断体检室是否可用.

相应的实现类也要修改:

使用 Moq 测试.NET Core 应用 -- Mock 属性

转会审批方法里也要修改:

使用 Moq 测试.NET Core 应用 -- Mock 属性

而在单元测试的方法里, 肯定是报错的:

使用 Moq 测试.NET Core 应用 -- Mock 属性

按照正常的思路, 我们可能会这样做:

使用 Moq 测试.NET Core 应用 -- Mock 属性

就是从内到外一层一层的mock.

这么做是没问题的, 测试也会通过:

使用 Moq 测试.NET Core 应用 -- Mock 属性

但是这样做很麻烦, 而Moq则提供了一种简单的方式来处理这种多层的/递归的mock:

使用 Moq 测试.NET Core 应用 -- Mock 属性

这样写即可. 测试同样会通过:

使用 Moq 测试.NET Core 应用 -- Mock 属性

为属性设置默认值

但是, 问题来了, 我还有一些其它的单元测试方法, 它们也需要用到这个属性, 现在它们的状态是:

使用 Moq 测试.NET Core 应用 -- Mock 属性

有的测试失败是因为其MockBehavior是Strict的, 而其它的失败则是因为里面出现了NullReferenceException.

针对这些情况, 我们可以这样设定:

使用 Moq 测试.NET Core 应用 -- Mock 属性

这样设置之后, 它会返回属性类型的默认值, 因为我没有设定返回值.

虽然测试依然不通过, 这是因为逻辑上的问题, 而不会抛出异常:

使用 Moq 测试.NET Core 应用 -- Mock 属性

针对这种情况, 还有一种更好的办法. 我们可以为mock对象设定默认值:

使用 Moq 测试.NET Core 应用 -- Mock 属性

把DefaultValue的值设为DefaultValue.Mock.

但是DefaultValue这个属性只对引用类型起作用(对值类型不起作用), 像这种递归的mock, 它会递归的创建所需的引用类型, 但是最后的IsAvailable这个值类型是不起作用的.

测试:

使用 Moq 测试.NET Core 应用 -- Mock 属性

因为最后一层是bool类型的, 是值类型, 所以上面的设置不起作用, 返回的是false. 所以测试没通过.

那我就把它改成string类型好了:

使用 Moq 测试.NET Core 应用 -- Mock 属性

审批方法:

使用 Moq 测试.NET Core 应用 -- Mock 属性

然后再调试测试:

使用 Moq 测试.NET Core 应用 -- Mock 属性

string是引用类型, 但是mock的值依然是null...??!!??

这是因为string是一个sealed class, 而DefaultValue.Mock只对接口, 抽象类和非sealed的class起作用....

不过测试仍然是可以通过的, 因为我改逻辑了:

使用 Moq 测试.NET Core 应用 -- Mock 属性

注意, 这个默认值只对宽松(Loose) mock, 起作用.针对Strict mock, 仍然需要设定最后一层属性的值.

属性值变化跟踪

需要添加一些代码, 首先添加一个枚举:

使用 Moq 测试.NET Core 应用 -- Mock 属性

为接口添加属性:

使用 Moq 测试.NET Core 应用 -- Mock 属性

实现类:

使用 Moq 测试.NET Core 应用 -- Mock 属性

然后在审批类里, 我设置了这个属性的值:

使用 Moq 测试.NET Core 应用 -- Mock 属性

上面的代码也就是说, 我的mock对象的某个属性在测试的时候它的值会发生变化. 而Moq可以记住这些mock属性的变化的值.....

新写一个测试:

使用 Moq 测试.NET Core 应用 -- Mock 属性

这里使用mockObj.SetupProperty()方法来开始追踪属性. 这个测试会通过:

使用 Moq 测试.NET Core 应用 -- Mock 属性

该方法也可以通过下面的写法来为被追踪的属性设置默认值:

mockExamination.SetupProperty(x => x.PhysicalGrade, PhysicalGrade.Failed);.

如果这个对象上有很多属性需要进行设置和追踪, 那么可以使用:

mock.SetupAllProperties(); 这个方法:

使用 Moq 测试.NET Core 应用 -- Mock 属性

注意, 这个方法应该最先调用, 否则的话其它的设置可能会被覆盖.

本文完成的代码在: https://github.com/solenovex/Moq4-Tutorial-Code 里面的03 After.

未完待续......