RhinoMock入门(7)——Do,With和Record-playback

简介: (一)Do(delegate) 有时候在测试过程中只返回一个静态的值是不够的,在这种情况下,Do()方法可以用来在方法调用时添加自定义的行为。一般来说,Do()方法会替换方法调用。它的返回值会从模拟的调用中返回(即使是有异常发生也是这样)。

(一)Do(delegate)

有时候在测试过程中只返回一个静态的值是不够的,在这种情况下,Do()方法可以用来在方法调用时添加自定义的行为。一般来说,Do()方法会替换方法调用。它的返回值会从模拟的调用中返回(即使是有异常发生也是这样)。Do()的参数委托委托的方法的签名须和方法的签名匹配。只有当签名匹配时才能生效,且一个匹配生效一次。

看官方给出的例子:

public   class  Speaker
{
    
private   readonly   string  firstName;
    
private   readonly   string  surname;
    
private  INameSource nameSource;

    
public  Speaker( string  firstName,  string  surname, INameSource nameSource)
    {
        
this .firstName  =  firstName;
        
this .surname  =  surname;
        
this .nameSource  =  nameSource;
    }

    
public   string  Introduce()
    {
        
string  name  =  nameSource.CreateName(firstName, surname);
        
return   string .Format( " Hi, my name is {0} " , name);
    }
}

  

public   interface  INameSource
{
    
string  CreateName( string  firstName,  string  surname);
}

 

现在演讲者和名字分开在两个类中。然后进行自我介绍,介绍时要介绍自己的姓名,即 FirstName+LastName 。在介绍中要用到 InameSource 中的 CreateName 方法,接下来将会模拟这个接口,而通过其它的方法来替代。

 

[Test]
public   void  SayHelloWorld()
{
    MockRepository mocks 
=   new  MockRepository();
    INameSource nameSource 
=  mocks.DynamicMock < INameSource > ();

    Expect.Call(nameSource.CreateName(
null null ))
          .IgnoreArguments()
          .Do(
new  NameSourceDelegate(Formal));

    mocks.ReplayAll();
    
string  expected  =   " Hi, my name is Ayende Rahien " ;
    
string  actual  =   new  Speaker( " Ayende " " Rahien " , nameSource).Introduce();
    Assert.AreEqual(expected, actual);
}

delegate   string  NameSourceDelegate( string  first,  string  surname); 

private   string  Formal( string  first,  string  surname)
{
    
return  first  +   "   "   +  surname;
}

 

看上段测试的粗体部分。

Do参数是委托类型,其中这个委托类型委托的方法的签名要和模拟对象中期望的要替换的方法的签名一致,即:

private   string  Formal( string  first,  string  surname)
string  CreateName( string  firstName,  string  surname);

 

两者相匹配。

然后当对演讲者构造时:
new Speaker("Ayende", "Rahien", nameSource)

会对演讲者三个域进行赋值

private   readonly   string  firstName;
private   readonly   string  surname;
private  INameSource nameSource;

 

接下来进行介绍时,调用方法:

public   string  Introduce()
{
    
string  name  =  nameSource.CreateName(firstName, surname);
    
return   string .Format( " Hi, my name is {0} " , name);
}

 

而这个方法则由Do方法的委托参数委托的方法来替代:

private   string  Formal( string  first,  string  surname)
{
    
return  first  +   "   "   +  surname;
}

 

返回FirstName+空格+LastName

 

(二)With

流畅式的期望和验证语法。什么是流畅式?先看例子:

[Test]
public   void  TestFluent()
{
    MockRepository mocks 
=   new  MockRepository();
    var customer 
=  mocks.DynamicMock < ICustomer > ();

    
string  strTemp = string .Empty; 

    With.Mocks(mocks)
        .Expecting(
          
delegate
          {
             Expect.Call(customer.ShowTitle(
"" )).Return( " with 语句 " );
          })
        .Verify(
         
delegate
          {
             strTemp 
=  customer.ShowTitle( "" );
          });

    Assert.AreEqual(
" with 语句 " , strTemp);
}

 

这就是所谓的流畅式。通过匿名委托来实现。如果在匿名委托中完成则会隐匿调用ReplayAll()mocks.VerifyAll()

如果要启用次序,则可使用:ExpectingInSameOrder,例如:

[Test]
public   void  TestFluent()
{
    MockRepository mocks 
=   new  MockRepository();
    var customer 
=  mocks.DynamicMock < ICustomer > ();
    
string  strTemp = string .Empty;
    With.Mocks(mocks).ExpectingInSameOrder(
        
delegate
        {
           Expect.Call(customer.ShowTitle(
"" )).Return( " with 语句 " );
           Expect.Call(customer.Unid).Return(
1 );
        })
       .Verify(
       
delegate
       {
          strTemp 
=  customer.ShowTitle( "" );
          
int  i  =  customer.Unid;
       });

    Assert.AreEqual(
" with 语句 " , strTemp);
}

 

With语句的隐式使用

With可以隐式的创建Mock实例,并自动调用VerifyAll方法。

[Test]
public   void  TestWithMocker()
{
    With.Mocks(
        
delegate
        {
          var customer 
=  Mocker.Current.DynamicMock < ICustomer > ();
          Expect.Call(customer.ShowTitle(
"" )).Return( " with 语句 " );
          Mocker.Current.ReplayAll();

          Assert.AreEqual(
" with 语句 " , customer.ShowTitle( "" ));
        });
}

 

这里才看出With确实很流畅。

下边说一下由显式创建Mock实例来代替隐式创建:

[Test]
public   void  TestWithMockerr()
{
    MockRepository mocks 
=   new  MockRepository();
    With.Mocks(mocks,
       
delegate
       {
          var customer 
=  Mocker.Current.DynamicMock < ICustomer > ();
          Expect.Call(customer.ShowTitle(
"" )).Return( " with 语句 " );

          Mocker.Current.ReplayAll();

          Assert.AreEqual(
" with 语句 " , customer.ShowTitle( "" ));
       });
}

 

没多大区别。在使用Mocker.Current时,不能在嵌套中使用,因为这是个全局的,而With.Mocks会重写Mocker.Current

(三)Record-PlayBack

Rhinomock支持一种通过Using语句来进行录制回放的方式。

[Test]
public   void  TestRecordPlayback()
{
    MockRepository mocks 
=   new  MockRepository();
    var customer 
=  mocks.DynamicMock < ICustomer > ();

    
using  (mocks.Record())
    {
        Expect.Call(customer.ShowTitle(
"" )).Return( " 录制回放 " );
    }

    
using  (mocks.Playback())
    {
        Assert.AreEqual(
" 录制回放 " , customer.ShowTitle( "" ));
    }
}

 

 

博客园大道至简

http://www.cnblogs.com/jams742003/

转载请注明:博客园

目录
相关文章
|
2月前
|
Java
Optional源码分析(涉及Objects源码和Stream源码)
本文分析了Java中Optional类的源码,包括其内部的Objects.requireNonNull方法、EMPTY定义、构造方法、ofNullable方法、isEmpty方法以及如何与Stream类交互,展示了Optional类如何避免空指针异常并提供流式操作。
41 0
Optional源码分析(涉及Objects源码和Stream源码)
|
6月前
|
安全 编译器 C#
Record
【6月更文挑战第11天】
55 3
|
7月前
|
存储 C++ 容器
C++进阶-- map和set
C++进阶-- map和set
|
7月前
|
SQL 数据挖掘 数据处理
「SQL面试题库」 No_122 Fix Product Name Format
「SQL面试题库」 No_122 Fix Product Name Format
|
存储
Leetcode Single Number II (面试题推荐)
给你一个整数数组,每个元素出现了三次,但只有一个元素出现了一次,让你找出这个数,要求线性的时间复杂度,不使用额外空间。
42 0
|
存储 自然语言处理 Go
C++进阶 map和set
C++进阶 map和set
124 0
C++进阶 map和set
|
SQL 存储 缓存
【MySQL从入门到精通】【高级篇】(二十四)EXPLAIN中select_type,partition,type,key,key_len字段的剖析
上一篇文章我们介绍了【MySQL从入门到精通】【高级篇】(二十三)EXPLAIN的概述与table,id字段的剖析,重点对EXPLAIN命令进行了阐述,并且对table,id字段进行了剖析。这篇文章接着对EXPLAIN命令的其余字段进行解析,本文将介绍select_type,partition,type,key,key_len 字段的含义。其中:读者朋友们需要重点掌握 select_type,type 两个字段的含义。
261 0
【MySQL从入门到精通】【高级篇】(二十四)EXPLAIN中select_type,partition,type,key,key_len字段的剖析
|
Java API
JAVA文档并发演示---并发高级对象摘录(Tutorial - Concurrency Lesson--High Level Concurrency Objects)
JAVA文档并发演示---并发高级对象摘录(Tutorial - Concurrency Lesson--High Level Concurrency Objects)
JAVA文档并发演示---并发高级对象摘录(Tutorial - Concurrency Lesson--High Level Concurrency Objects)
|
存储 JavaScript 前端开发
深入解析 Category 的实现原理
无论一个类设计的多么完美,在未来的需求演进中,都有可能会碰到一些无法预测的情况。那怎么扩展已有的类呢?一般而言,继承和组合是不错的选择。但是在Objective-C 2.0中,又提供了category这个语言特性,可以动态地为已有类添加新行为。如今category已经遍布于Objective-C代码的各个角落,从Apple官方的framework到各个开源框架,从功能繁复的大型APP到简单的应用,catagory无处不在。本文对category做了比较全面的整理,希望对读者有所裨益。
160 0
|
SQL 索引
【笔记】开发指南—DAL语句—CHECK GLOBAL INDEX
您可以使用CHECK GLOBAL INDEX语句检查主表和索引表的数据是否完全一致,并修订不一致的数据。