NSubstitute完全手册(十七)参数匹配器上的操作

简介:

除了指定调用,当替代实例接收到了匹配的调用时,参数匹配器还可以用于执行特定的操作,并指定操作参数。这是一个相当罕见的需求,但在某些情况下可以使测试程序变得简单一些。

警告:一旦我们为替代实例添加有意义的行为,我们的测试和代码将面临过度指定和紧耦合的风险。对于类似的测试,通过抽象来更好地封装行为可能是更好的选择,甚至可以使用真实的协作对象并切换至 coarser-grained 测试。

调用回调函数

假设被测试类需要调用一个依赖对象的方法,并为其提供了一个回调函数,当依赖对象调用结束时通过回调来通知该类。当替代实例被调用时,我们可以使用 Arg.Invoke() 来立即调用这个回调。

让我们来看一个例子。比如说我们要测试 OrderPlacedCommand,其需要使用 IOrderProcessor 来处理订单,当处理成功地完成时使用 IEvents 来引发事件通知。

复制代码
 1     public interface IEvents
 2     {
 3       void RaiseOrderProcessed(int orderId);
 4     }
 5 
 6     public interface ICart
 7     {
 8       int OrderId { get; set; }
 9     }
10 
11     public interface IOrderProcessor
12     {
13       void ProcessOrder(int orderId, Action<bool> orderProcessed);
14     }
15 
16     public class OrderPlacedCommand
17     {
18       IOrderProcessor orderProcessor;
19       IEvents events;
20       public OrderPlacedCommand(IOrderProcessor orderProcessor, IEvents events)
21       {
22         this.orderProcessor = orderProcessor;
23         this.events = events;
24       }
25       public void Execute(ICart cart)
26       {
27         orderProcessor.ProcessOrder(
28             cart.OrderId,
29             wasOk => { if (wasOk) events.RaiseOrderProcessed(cart.OrderId); }
30         );
31       }
32     }
复制代码

在测试中,可以使用 Arg.Invoke() 来模拟 IOrderProcessor 处理订单结束的情况,并调用回调函数来通知调用方已经处理结束。

复制代码
 1     [TestMethod]
 2     public void Test_ActionsWithArgumentMatchers_InvokingCallbacks()
 3     {
 4       // Arrange
 5       var cart = Substitute.For<ICart>();
 6       var events = Substitute.For<IEvents>();
 7       var processor = Substitute.For<IOrderProcessor>();
 8       cart.OrderId = 3;
 9       // 设置 processor 当处理订单ID为3时,调用回调函数,参数为true
10       processor.ProcessOrder(3, Arg.Invoke(true));
11 
12       // Act
13       var command = new OrderPlacedCommand(processor, events);
14       command.Execute(cart);
15 
16       // Assert
17       events.Received().RaiseOrderProcessed(3);
18     }
复制代码

这里我们构造了 processor,用于处理 ID 为 3 的订单,并调用回调函数。我们使用 Arg.Invoke(true) 来传递 true 给回调函数。

Arg.Invoke 有几个重载方法,可以用于调用参数数量和类型不同的回调函数。我们也可以使用 Arg.InvokeDelegate 来调用定制的委托类型(那些不只是简单的 Action 类型的委托)。

执行带参数的操作

有时我们可能不想立即就调用回调函数。或者可能我们想存储所有的实例到一个特殊的参数,并将该参数传递给一个方法。甚至我们只是想捕获某个参数,以便后期查看。我们可以使用 Arg.Do 来完成这些目的。Arg.Do 会为每个匹配的调用来执行给定参数的操作。

复制代码
 1     public interface ICalculator
 2     {
 3       int Multiply(int a, int b);
 4     }
 5 
 6     [TestMethod]
 7     public void Test_ActionsWithArgumentMatchers_PerformingActionsWithArgs()
 8     {
 9       var calculator = Substitute.For<ICalculator>();
10 
11       var argumentUsed = 0;
12       calculator.Multiply(Arg.Any<int>(), Arg.Do<int>(x => argumentUsed = x));
13 
14       calculator.Multiply(123, 42);
15 
16       Assert.AreEqual(42, argumentUsed);
17     }
复制代码

这里,Multiply 方法的第一个参数可为任意值,第二个参数将被传递值一个 argumentUsed 变量。当我们想断言某参数的多个属性时,这个功能是非常有用的。

复制代码
 1     [TestMethod]
 2     public void Test_ActionsWithArgumentMatchers_PerformingActionsWithAnyArgs()
 3     {
 4       var calculator = Substitute.For<ICalculator>();
 5 
 6       var firstArgsBeingMultiplied = new List<int>();
 7       calculator.Multiply(Arg.Do<int>(x => firstArgsBeingMultiplied.Add(x)), 10);
 8 
 9       calculator.Multiply(2, 10);
10       calculator.Multiply(5, 10);
11 
12       // 由于第二个参数不为10,所以不会被 Arg.Do 匹配
13       calculator.Multiply(7, 4567); 
14 
15       CollectionAssert.AreEqual(firstArgsBeingMultiplied, new[] { 2, 5 });
16     }
复制代码

在此例中,当调用 Multiply 方法第二个参数是 10 时,不管第一个 int 类型的参数的值是多少,Arg.Do 都会将其添加到一个列表当中。

参数操作和调用指定参数

就像 Arg.Any<T>() 参数匹配器一样,当参数的类型为 T 时,参数操作会调用一个指定操作(所以也可以用于设置返回值检查接收到的调用)。它只是多了个能与匹配指定规格的调用的参数交互的功能。

复制代码
 1     [TestMethod]
 2     public void Test_ActionsWithArgumentMatchers_ArgActionsCallSpec()
 3     {
 4       var calculator = Substitute.For<ICalculator>();
 5 
 6       var numberOfCallsWhereFirstArgIsLessThan0 = 0;
 7 
 8       // 指定调用参数:
 9       // 第一个参数小于0
10       // 第二个参数可以为任意的int类型值
11       // 当此满足此规格时,为计数器加1。
12       calculator
13         .Multiply(
14           Arg.Is<int>(x => x < 0),
15           Arg.Do<int>(x => numberOfCallsWhereFirstArgIsLessThan0++)
16         ).Returns(123);
17 
18       var results = new[] {
19         calculator.Multiply(-4, 3),
20         calculator.Multiply(-27, 88),
21         calculator.Multiply(-7, 8),
22         calculator.Multiply(123, 2) // 第一个参数大于0,所以不会被匹配
23       };
24 
25       Assert.AreEqual(3, numberOfCallsWhereFirstArgIsLessThan0); // 4个调用中有3个匹配上
26       CollectionAssert.AreEqual(results, new[] { 123, 123, 123, 0 }); // 最后一个未匹配
27     }
复制代码

NSubstitute 完全手册






本文转自匠心十年博客园博客,原文链接:http://www.cnblogs.com/gaochundong/archive/2013/05/22/nsubstitute_actions_with_argument_matchers.html,如需转载请自行联系原作者
目录
相关文章
|
测试技术 编译器 数据库
【软件设计师备考 专题 】软件开发环境和工具:从分析到维护(一)
【软件设计师备考 专题 】软件开发环境和工具:从分析到维护
268 0
|
缓存 Linux
centos7更换yum源 一条命令搞定
centos7更换yum源 一条命令搞定
2619 0
|
7月前
|
SQL 消息中间件 Serverless
​Flink+Paimon+Hologres,面向未来的一体化实时湖仓平台架构设计
​Flink+Paimon+Hologres,面向未来的一体化实时湖仓平台架构设计
182 4
Win环境下安装 torch==1.1.0
Win环境下安装 torch==1.1.0
279 2
|
监控 数据可视化 调度
ERP系统中的生产排程与生产效率分析解析
【7月更文挑战第25天】 ERP系统中的生产排程与生产效率分析解析
430 0
|
机器学习/深度学习 人工智能 自然语言处理
|
敏捷开发 数据管理 测试技术
深入探索软件测试:方法、挑战与最佳实践
【2月更文挑战第20天】 在数字化时代,软件已成为不可或缺的核心。确保软件的质量和可靠性,软件测试扮演着至关重要的角色。本文将深入探讨软件测试的各种方法,面临的挑战以及实施的最佳实践。我们将从单元测试的基础出发,逐步过渡到集成和系统测试,最后讨论自动化测试的策略。文章旨在为软件测试工程师提供深度洞见,帮助他们更有效地设计和执行测试计划,以确保软件产品的质量。
|
Kubernetes 容灾 Java
nacos常见问题之启动报错如何解决
Nacos是阿里云开源的服务发现和配置管理平台,用于构建动态微服务应用架构;本汇总针对Nacos在实际应用中用户常遇到的问题进行了归纳和解答,旨在帮助开发者和运维人员高效解决使用Nacos时的各类疑难杂症。
|
存储 Kubernetes 前端开发
容器服务ACK常见问题之把容器的时间改成宿主机的时区失败如何解决
容器服务ACK(阿里云容器服务 Kubernetes 版)是阿里云提供的一种托管式Kubernetes服务,帮助用户轻松使用Kubernetes进行应用部署、管理和扩展。本汇总收集了容器服务ACK使用中的常见问题及答案,包括集群管理、应用部署、服务访问、网络配置、存储使用、安全保障等方面,旨在帮助用户快速解决使用过程中遇到的难题,提升容器管理和运维效率。
|
前端开发 JavaScript 容器
【Vue2.0】—解决页面闪烁的问题(八)
【Vue2.0】—解决页面闪烁的问题(八)