【Pro ASP.NET MVC 3 Framework】.学习笔记.4.MVC的主要工具-使用Moq

在之前的例子中,我们创建了FakeRepository类来支持我们的测试。但是我们还没有解释如何穿件一个真实的repository实现,我们需要一个替代品。一旦我们有一个真的实现,我们可能不会再用它,因为它把我们的测试环境变得复杂。

FakeRepository类,是IProductRepository接口的伪实现。我们创建伪实现,并手动添加特别的参数,使得FakeRepository类手动的仿制品。Moq是一个框架,让我们仿制变得快速,简单,容易。

1 将Moq添加到测试项目,而不是应用程序项目

2 使用Moq创建一个Mock

使用mocking工具的好处是,我们能创建一个为满足测试中的功能定制的Mocks。这意味着我们最终不会得到太复杂的mock实现。在一个真实的项目中,不像这些简单的例子。我们能容易地抵达舞台,mock实例需要它自己的测试,因为它包含足够的代码。我们能手工创建一些mock,为了使它们生效,我们需要移动循环代码到基类,我们正确的返回会变得更复杂。有两个舞台需要使用Moq创建mock,第一个创建一个新的Mock<T>,这个T是我们想要mock的。

第二个舞台是配置实现要展示的行为。Moq会自动实现我们在类型中给它的所有的方法和属性,它会使用类型的默认值。例如,IProductRepository.GetProducts方法会返回一个空的IEnumerable<Product>。要改变Moq实现一个类型成员的方法,我们需要使用Setup方法。

3 使用Moq方法选择器

第一个参数是被选择的方法。Moq使用Linq和Lambda表达式。当我们调用Setup方法,Moq传递接口。当我们想要为GetProducts方法定义一个行为,我们可以这样做

1 mock.Setup(m => m.GetProducts()).(<...other methods...>);

我们不需要知道它内部是怎样工作的,只需要照着做就行了。GetProduct方法容易处理,是因为它没有参数。如果我们想要处理携带参数的方法,我们需要考虑第二个元素:参数过滤。

4 使用Moq的参数渗透

1 publicinterface IMyInterface { 2 string ProcessMessage(string message); 3 } 4 5 Mock<IMyInterface> mock =new Mock<IMyInterface>(); 6 mock.Setup(m => m.ProcessMessage("hello")).Returns("Hi there"); 7 mock.Setup(m => m.ProcessMessage("bye")).Returns("See you soon");

要为所有可能的参数设置相应,可以使用Moq提供的It类。

1 mock.Setup(m => m.ProcessMessage(It.IsAny<string>())).Returns("Message received");

It类定义一些方法,配合一般型参数使用。我们调用IsAny方法,使用string作为一般类型。这告诉Moq,当ProcessMessage方法伴着任何string值被调用,它会返回相应Message Recived。

Method Description
Is<T>() 匹配基于指定的条件
IsAny<T>() 当参数是任何T类型的实例时匹配
IsInRange<T> 当参数在指定值之间时匹配
IsRegex 当匹配指定的正则表达式时匹配

Is<T>方法时最灵活的,因为它让我们提供一个条件。

1 mock.Setup(m => m.ProcessMessage(It.Is<string>(s => s =="hello"|| s =="bye"))) 2 .Returns("Message received");

当string参数是hello或bye时,返回Message Recived。

5 返回一个结果

当我们配置行为时,我们也定义它被触发时的返回方法。上个例子中,Returns方法链式地Setup方法。我们也可以使用传递给mocked方法的参数,给Return方法,让output基于input。

1 mock.Setup(m => m.ProcessMessage(It.IsAny<string>())) 2 .Returns<string>(s =>string.Format("Message received: {0}", s));

6 使用Moq的单元测试

一旦配置好必须的行为,你可以通过Mock.Object属性得到mocked的实现。

1 [TestClass()] 2 publicclass MyPriceReducerTest 3 { 4 private IEnumerable<Product> products; 5 6 [TestInitialize] 7 publicvoid PreTestInitialize() 8 { 9 products =new Product[] 10 { 11 new Product(){Name="Kayak",Price=275M}, 12 new Product(){Name="Lifejacket",Price=48.95M}, 13 new Product(){Name="Soccer ball",Price=19.50M}, 14 new Product(){Name="Stadium",Price=79500M} 15 }; 16 }

为所有测试准备公共数据。单元测试的属性:

Attribute Description
ClassInitialize 在类中的单元测试被执行之前调用。必须应用给静态方法
ClassCleanup 在类中的所有方法执行完成后调用。必须应用给静态方法
TestInitialize 在每个测试执行前调用
TestCleanup 在每个测试执行后调用

VS只看这些属性,方法的名字不重要。

7 使用Moq验证

当每个Product对象被处理时,UpdateProduct方法会被调用。在FakeRepository类中,我们我们定义了一个自增的属性。我们能用Moq以更优美的方式实现相同的效果。

1 //Assert2 foreach (Product p in products) 3 { 4 mock.Verify(m => m.UpdateProduct(p), Times.Once()); 5 }

使用参数渗透,我们能验证UpdateProduct方法,恰好被每个Product对象调用一次。

posted @ 2013-08-26 09:30  Reinhard_Hsu  阅读(306)  评论(0编辑  收藏  举报