博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
ASP.Net MVC3 - The easier to run Unit Tests by moq #Reprinted#
阅读量:4356 次
发布时间:2019-06-07

本文共 5718 字,大约阅读时间需要 19 分钟。

From: http://www.cnblogs.com/techborther/archive/2012/01/10/2317998.html

 


 

前几天调查完了unity。现在给我的任务是让我调查Moq。

以下是自己找了资料,总结并实践的内容。如果有表述和理解错误的地方。恳请指正。

什么是Moq?

 

Moq(英语发音是Mock-you 或者只是mock)是一个针对.Net开发的模拟库,它从开始就完全充分利用了.NET3.5(LINQ表达式树)和C#3.0的新特性(lambda表达式)。它的目标是让模拟以一种自然的方式与现有单元测试进行集成,使它更加简单、直观,以避免开发人员被迫重写测试或高成本的学习测试框架。这使它成为了一个高生产力、类型安全、重构友好的模拟库。

从哪得到Moq?

 

如果你看过我的其他文章,我们可以直接使用 VS中的插件Nuget来获取Moq并且引用到指定的项目。

否则,我们可以从这里得到Moq的最新版本。

可以模拟什么?局限性

 

首先,模拟的类不能是密封的。

其次,你不能直接模拟静态方法。因为Moq只能创建模拟对象实例。在这种情况下,间接的解决方案是我们可以在要模拟对象外包装一层,并且去模拟这个新对象。这种模式被称为适配器模式。

通常我们测试一个方法,它有可能调用好几个service。但是每次都去访问这些service的代价是很高的。我们可以通过模拟的方法让它模拟访问service,并且根据不同请求模拟返回响应的结果。

Moq原理

 

Moq是如何办到的?它只需要一个接口类型就可以生产一个对象?没错,就是这样。Moq使用  完成这个任务。基本原理就是它利用反射机制的  功能动态生成一个空类型(也就是所有接口的方法都实例化,但是没有任何功能,只是一个程序骨架)。所以Mock的能力就在于可以利用DynamicProxy的机制快速生产出一个假对象来,用于模仿真对象的行为。

 

Moq中的重要成员

 

Mock

通过这个类,我们可以得到一个Mock<T>对象。T可以是接口,也可以是类。它有一个public 和virtual属性。让我们看看下边的例子:

//define interface to be mocked        public interface IFake        {            bool DoSomething(string actionname);        }        //define the test method        [TestMethod]        public void Test_Interface_IFake()        {            //make a mock Object by Moq            var mo = new Mock
(); //Setup our mock object mo.Setup(foo => foo.DoSomething("Ping")) .Returns(true); //Assert it! Assert.AreEqual(true, mo.Object.DoSomething("Ping")); }

 

在上边的代码,我们通过传递泛型参数IFake去创建Mock<IFake>的实例 模拟接口IFake。

接下来我们要调用Setup()方法去创建我们的模拟对象。注意,Setup方法的参数是一个lambda表达式。我们可以这样理解:当被模拟的对象foo调用它自己的方法DoSomething(),并且参数是Ping。添加后缀 Return (true)我们可以理解为:前边的请求返回结果为真。这是我们指定的返回值。当一个请求调用DoSomething()方法时。如果传入的参数是Ping,那么我们会返回true。接下来,我们添加一个断言,去判断是否能得到预期结果。

注:Foo仅仅是一个词用作通用替代真实的东西,特别是在讨论技术想法和问题.

It

 

这是一个静态类,定义了静态的泛型方法:Is<TValue>, IsAny<TValue>, IsInRange<TValue>, 和IsRegex。去过滤参数。看看下边例子:

public interface IEmailSender        {            bool Send(string subject, string body, string email);        }  [TestMethod]        public void User_Can_Send_Password()        {            var emailMock = new Mock
(); emailMock .Setup(sender => sender.Send(It.IsAny
(), It.IsAny
(), It.IsAny
())) .Returns(true); }

 

任何时候调用Send()方法,只要传入的参数是任何的string,我们定义他会返回true。

我们也可以根据lambda的优势订制一个规则:

var productRepository = new Mock
(); productRepository .Expect(p => p.Get(It.Is
(id => id > 0 && id < 6))) .Returns(newProduct.Object);

 

这样我们可以设置这个id在0和6之间的时候才会返回一个新的对象。上边提及到的其他方法,我们可以参考这里的教程 

此外,这个类的增强版是 Match<T>,你完全可以自定义模拟规则。

MockBehavior

 

这个类用于模拟对象的行为。就像是否按照默认的模式去模拟。让我们进一步看看他的定义:

namespace Moq{    // Summary:    //     Options to customize the behavior of the mock.    public enum MockBehavior    {        // Summary:        //     Causes the mock to always throw an exception for invocations that don't have        //     a corresponding setup.        Strict = 0,        //        // Summary:        //     Will never throw exceptions, returning default values when necessary (null        //     for reference types, zero for value types or empty enumerables and arrays).        Loose = 1,        //        // Summary:        //     Default mock behavior, which equals Moq.MockBehavior.Loose.        Default = 1,    }}

 

现在,看看如下例子:

 

var mock = new Mock
(MockBehavior.Strict);

指定了mock行为是精准的,如果没有按照预期的Setup就会抛出异常。

MockFactory

 

这是一个模拟对象的工厂,我们不仅仅可以定制创建模拟对象的配置,也可以成批测试它们。看看下边例子:

var factory = new MockFactory(MockBehavior.Strict) { DefaultValue = DefaultValue.Mock };            // Create a mock using the factory settings            var fooMock = factory.Create
(); // Create a mock overriding the factory settings var barMock = factory.Create
(MockBehavior.Loose); // Verify all verifiable expectations on all mocks created through the factory factory.Verify();

 

在前边,我们已经介绍了传统方式的创建模拟对象,它不会去真正的调用方法,而是仅仅去执行一些假设:如果...那么返回... 。

在下边的例子里,我将继续介绍一些Mock<T> 类中基本并且重要的方法。

Verification

 

有时候,我们要确定一个方法是否被调用了,或者甚至要知道它被调用了多少次。一个比较传统的方式是使用Verify()方法。看看下边例子:

mock.Verify(foo => foo.DoSomething("Ping"), Times.Once());

上边的代码尝试验证DoSomething("Ping")需要被调用,并且只调用一次。出了Once选项,这里也有更多的选项可供你选择去决定这个方法需要被调用多少次。如: AtLeast, AtLeastOnce, AtMost, AtMostOnce, Between, Equals, Exactly, Never, 和Once

一旦我们已经模拟了对象,验证将是个轻松的任务,看看下边的例子:

[TestMethod]        public void Test_FindByName_GetCalled()        {            // create some mock data            IList
products = new List
{ new Product { ProductId = 1, Name = "C# Unleashed", Description = "Short description here", Price = 49.99 }, new Product { ProductId = 2, Name = "ASP.Net Unleashed", Description = "Short description here", Price = 59.99 }, new Product { ProductId = 3, Name = "Silverlight Unleashed", Description = "Short description here", Price = 29.99 }}; Mock
mock = new Mock
(); //mock //.Setup(sender => sender.FindById(It.IsAny
())) //.Returns((int s) => products.Where( // x => x.ProductId == s).Single()); mock.Object.FindById(1); mock .Verify(x => x.FindById(1), Times.Once()); } }

 

在上边的例子里,有两个地方值得注意。

首先是mock.Object.FindById(1)。为了在这个case里让一切变得简单,我们直接调用mock.Object的方法。为什么呢?因为我们这个case只关注这个方法被调用的次数,而不关注返回值。当然,在实际的应用中我们很少这样做。

第二,你有没有注意到被注释掉的句子。它仅仅是一个“如果,那么”句子。意思是说,如果Setup 好了,就返回我们定义的值。

由于调查的时间有限,还有很多关于Mock给力的地方我没有涉及到。我们可以去查看官方的资料。欢迎共同讨论。

 

 

参考资料

转载于:https://www.cnblogs.com/wushuaiyi/p/4667844.html

你可能感兴趣的文章
LPC43xx SGPIO I2C Implementation
查看>>
Beta 冲刺(1/7)
查看>>
修改 Vultr 登录密码
查看>>
CSS学习
查看>>
Centos 安装lnmp完整版
查看>>
PHP把图片存入数据库(非路径)【待测试】
查看>>
ZH奶酪:PHP判断图片格式的7种方法
查看>>
java中给main传参的方式
查看>>
Git常用
查看>>
springboot实现邮件发送
查看>>
Python3.x:抢票
查看>>
前端三大主流框架的对比React、Vue、Angular 所谓是是三分天下
查看>>
python高阶函数
查看>>
浅谈C#Socket
查看>>
设计模式之策略模式
查看>>
C++学习笔记(七)——函数缺省参数值
查看>>
UVA - 10003 Cutting Sticks
查看>>
JS 弹出网页 (不显示地址栏,工具栏) 网页去掉地址栏
查看>>
使用update包更新系统文件的过程
查看>>
js高级程序设计--DOM
查看>>