Moq 和RhinoMocks

Moq & RhinoMocks

使用Mock对象进行测试一般都会有以下三个关键步骤:

  1. 使用接口来描述需要测试的对象
  2. 为实际的产品代码实现这个接口
  3. 以测试为目的,在Mock对象中实现这个接口

在使用Mock对象的过程中,充分体现出了“面向接口编程”的设计原则,同时也促成类的良好设计。

自行实现Mock对象是相当繁琐的工作,让人幸运的是,在.NET世界中有多个优秀的Mock框架可以供大家选择,目前最常使用的无非MoqRhino Mocks这两个框架。两者的最新版本在Mocking API方面的用法已日趋一致,都依托Lambda表达式、泛型和扩展方法做了很大改进,目标都是让Mock对象以一种更自然的方式与多个单元测试框架进行集成,以一种清晰的语法来描述期望值、参数约束、返回值等,极大的方便开发者的使用。

由于Moq和Rhino Mocks都使用了Castle DynamicProxy这个类库动态生成代理类,因此对需要Mock的对象有一定的限制:所测试的方法必须是virtual类型。

下面就用一个例子来看看两者的不同实现(这个例子摘自Moq源代码包中的Samples,只是略做了些修改以便于展现两者的特点):

需要进行测试的对象如下示之:

展开

下面是用这两个Mock框架分别实现的单元测试代码:

Moq 4.0 Rhino Mocks 3.6
[Test]
public void TestPresenterSelection() {
    // arrange
    var mView = new Mock<IOrdersView>();
    var mRepository = new Mock<IRepository<Order>>();
    var presenter = new OrdersPresenter(mView.Object, mRepository.Object);
    // check that the presenter has no selection by default
    Assert.Null(presenter.SelectedOrder);

    // raise event
    mView.Raise(io => io.OrderSelected += null,
        new OrderEventArgs { Order = new Order("moq", 50) });

    // assert
    Assert.NotNull(presenter.SelectedOrder);
    Assert.AreEqual("moq", presenter.SelectedOrder.ProductName);
}
[Test]
public void TestPresenterSelection() {
    // arrange
    var mView = MockRepository.GenerateMock<IOrdersView>();
    var mRepository = MockRepository.GenerateMock<IRepository<Order>>();
    var presenter = new OrdersPresenter(mView, mRepository);
    // check that the presenter has no selection by default
    Assert.Null(presenter.SelectedOrder);

    // raise event
    mView.Raise(io => io.OrderSelected += null, null,
        new OrderEventArgs { Order = new Order("moq", 50) });

    // assert
    Assert.NotNull(presenter.SelectedOrder);
    Assert.AreEqual("moq", presenter.SelectedOrder.ProductName);
}
[Test]
public void TestRetrieveOrders() {
    // arrange
    var mView = new Mock<IOrdersView>();
    var mRepository = new Mock<IRepository<Order>>();
    var presenter = new OrdersPresenter(mView.Object, mRepository.Object);

    List<Order> defaultOrders = new List<Order>
        { new Order("moq"), new Order("RhinoMock") };

    mRepository.Setup(r => r.FindAll()).Returns(defaultOrders);

    // exercise mocks
    presenter.OnInit();
    // assert
    mView.VerifySet(v => v.Orders = defaultOrders);
}
[Test]
public void TestRetrieveOrders() {
    // arrange
    var mView = MockRepository.GenerateMock<IOrdersView>();
    var mRepository = MockRepository.GenerateStub<IRepository<Order>>();
    var presenter = new OrdersPresenter(mView, mRepository);

    List<Order> defaultOrders = new List<Order>
        { new Order("moq"), new Order("RhinoMock") };

    mRepository.Stub(ir => ir.FindAll()).Return(defaultOrders);

    // exercise mocks
    presenter.OnInit();
    // assert
    mView.AssertWasCalled(v => v.Orders = defaultOrders);
}

Conclusion

通过上面的实例我们可以很容易看出两者的Syntax与API都非常接近,使用两者任何一个都能方便实现你的测试目的。

References

  1. QuickStart - Moq
  2. [email protected] - Rhino Mocks 3.5
  3. Daniel Cazzulino‘s Blog - Why do we need yet another NET mocking framework

分类: C#

标签: MoqRhino Mocks

时间: 2024-11-05 15:44:56

Moq 和RhinoMocks的相关文章

如何使用MOQ进行单元测试

使用MOQ来伪装和隔离被依赖对象,从而提高被测对象的测试效果. 安装 通过http://code.google.com/p/moq可以下载MOQ的最新版本.在SSL项目中,我们使用的是MOQ 3.1.416.3版本.在SCM中项目目录下的Lib目录下有该工具的二进制版本.直接在单元测试项目中引用即可. 准备工作 如果你需要测试项目中的Internal成员,你需要在AssemblyInfo.cs中添加如下的Attribute: #if DEBUG [assembly: InternalsVisib

【PRO ASP.NE MVC4 学习札记】使用Moq辅助进行单元测试

清楚问题所在: 先开个头,当我们对A进行单元测试时,可能会发现A的实现必须要依赖B.这时,我们在写单元测试时,就必须先创建B的实例,然后把B传给A再建立A的实例进行测试. 这样就会出现一些问题: 1.我们的单元测试会变得复杂而且脆弱.复杂是因为我们必须要花费精力去弄清楚B的逻辑.脆弱是因为如果B的逻辑更改了,我们对A的单元测试也可能会面临失败. 2.更重要的是,当我们测试失败时,我们无法很快定位到究竟是A除了问题还是B出了问题. 所以我们使用Moq这种技术来Mock “伪造” 一个B的实例,这样

使用Ninject+Moq在单元测试中抽象数据访问层

一.测试方法的业务逻辑时,通常都需要从数据库读取测试数据,但是每次初始化数据库数据都很麻烦,也会影响到其它业务对数据的访问,怎样抽象数据访问层呢?就是用Moq去模拟数据访问的逻辑 二.步骤如下 2.1 定义数据访问接口和实现 public interface IDBAccess { List<string> GetList(string request); } public class DBAccessImp : IDBAccess { public List<string> Ge

【MVC 4】4.MVC 基本工具(Visual Studio 的单元测试、使用Moq)

 作者:[美]Adam Freeman      来源:<精通ASP.NET MVC 4> 3.Visual Studio 的单元测试 有很多.NET单元测试包,其中很多是开源和免费的.本文打算使用 Visual Studio 附带的内建单元测试支持,但其他一些.NET单元测试包也是可用的. 为了演示Visual Studio的单元测试支持,本例打算对示例项目添加一个 IDiscountHelper 接口的新实现. 在 Models 文件夹下新建类文件 MinimumDiscountHelpe

单元测试——Moq框架下的Mock简介及简单实例

1.为什么要用Mock? 在测试过程中,往往会遇到这些问题: (1)真实对象具有不可确定的行为(产生不可预测的结果,如股票行情). (2)真实对象很难被创建. (3)真实对象的某些行为很难触发(如网络错误). (4)真实对象令程序的运行速度很慢. (5)真实对象有(或者是)用户界面. (6)测试需要询问真实对象它是如何被调用的(例如,测试可能需要验证某个回调函数是否被调用了). (7)真实对象实际上并不存在(当需要和其他开发小组,或者新的硬件系统打交道的时候,这是一个普遍问题). 借助于mock

单元测试与Moq

这个篇幅里面,记录单元测试与Moq模拟包的知识点. 单元测试 每一个模块,都应该有对应的单元测试.单元测试可以保证你的代码准确性,大大减少出现BUG的几率.一个好的单元测试,也是重构代码必不可少的部分. 我们可以直接在vs中,新增单元测试项目. 类库中,只要声明TestClass,就当做是单元测试. 单元测试分为三部分:准备,行为,断言.其中准备和行为,就是模拟程序代码执行.断言,就是判断程序执行结果与预期结果是否一致. 断言使用:Assert类来实现.详细参见:Assert类 [TestCla

单元测试 之 NUnit和RhinoMocks

在dotnet环境中,我们通常使用NUnit和RhinoMocks来编写单元测试. NUnit 简介 NUnit是.net平台上的一个单元测试框架,用来帮助开发人员方便的完成单元测试的编写.其主页为 http://www.nunit.org,目前最新版本为2.6.3. 可以通过以下链接来查看不同版本的使用文档 http://www.nunit.org/index.php?p=documentation Rhino Mocks简介 Rhino Mocks是一个模拟对象创建和管理框架.使用它,开发人

MOQ

MOQ:(Minimum order Quantity) 最低订货数量 MOQ 即最小订购量(最小订单量) 对每个产品设定建议订单量是补货的方法之一.另外要注意订单的有效性,这是由供应商制定的规则.比如说最小订单量.包装数量和方式等等.采购者的建议订单量是在这个基础上制定的. 如果订单量比较少不能满足最小订单量,供应商不会送货,反之订单量比较高,就可能导致较高的库存.这就需要采购者平衡两者的关系. 注意:最小订购量与供应商的成本紧密相关.供应商会根据生产情况与成本控制,根据客户订货量确定最小订单

Moq的使用

参考资料: 1. http://www.codeproject.com/Tips/729646/TDD-using-MOQ 2. https://github.com/Moq/moq4/wiki/Quickstart 从https://github.com/Moq/moq4下载编译,且编译需要依赖Castle.Core这个包