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,只是略做了些修改以便于展现两者的特点):

mocks 

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

展开

下面是用这两个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. Ayende@Wiki - Rhino Mocks 3.5
  3. Daniel Cazzulino's Blog - Why do we need yet another NET mocking framework

你可能感兴趣的:(mock)