单元测试之Mock

  • 为什么需要Mock.
    • 真实对象具有不确定的行为.所以会产生不可预测的结果.
    • 真实对象很难被创建.
    • 真实对象的某些行为很难被触发(如网络错误).
    • 真实对象令程序的运行速度很慢.
    • 真实对象有(或者是)用户界面.
    • 测试需要询问真实对象它是如何被调用的.
    • 真实对象实际上并不存在.例如其它小组开发的模块.
  • 使用Mock的3个步骤
    • 使用一个接口来描述该对象.
    • 为产品代码实现该接口.
    • 以测试为目的,在Mock对象中实现该接口.
  • Test Double
    • Dummy.被传递但是从不被实际使用的对象.通常用于填充参数列表.
    • Fake.含有实际实现的对象.但通常使用一些不适用于实际项目的捷径(内存数据库).
    • Stubs.对调用提供一些预设的响应.通常不会调用代码中实际调用会执行的代码.可以用来记录关于调用的信息.(Email中可以记录发送过的邮件).
    • Mocks.预先编写的对象.接受特殊的调用,返回特定的预期行为.
    • Mock的四个步骤:setup,exercise,verify,teardown.
    • 只有Mock是进行行为测试验证.
  • 主要对象和次要对象
    • 单元测试中,我们通常关注的是主要测试对象的功能和行为.
    • 对于主要测试对象涉及到的次要对象尤其是一些依赖,我们仅仅关注主要测试对象和次要测试对象的交互.
    • 比如是否调用,何时调用,调用的参数,调用的次序,以及返回的结果或者异常等.
    • 但是次要对象是如何执行这次调用的,并不关心.
    • 因此,使用mock对象或者stub对象来替代真实的次要对象,从而模拟真实场景来进行对主要测试对象的测试.
  • 测试工具
    • 在.Net下大部分工具都是使用动态类型来进行Mock,因此,只能Mock接口或者overriable成员.
    • 而TypeMock直接使用inject方式,即使是sealed或者不可覆盖方法也能进行Mock.
    • 通常,如果发现必须要覆盖不可覆盖的方法才能够进行测试,那么很可能是设计问题.
  • Mock行为依赖风险
    • 被模型对象的行为必须与真实对象的行为完全一致.
    • 开发者对API不够了解;被模拟对象的行为发生了改变(重构,添加新功能导致的).都可能引起错误假设(与真实对象行为不一致).错误假设会引入缺陷,并留下非法测试.
    • 非法测试:看起来像测试,运行起来也是测试.但是几乎没有价值,几乎不会失败.
  • Mock的优点
    • Mock对象的行为简单,唯一.一旦设置好setup后总是返回同一值.
    • Mock对象的行为可以预期,如果调用到了不希望调用的方法会让测试失败.若方法被调用了,还可以验证参数.
    • 可以Mock一些在真实环境中难以模拟或者出现的错误或者异常.
    • Mock是一种白盒测试方式.Mock对象的setup过程就是目标代码实现细节的设计过程.
    • 接口为使用者而设计.所以当接口还未被实现时,Mock可以验证使用者是如何使用接口的.
  • Mock的缺陷
    • Mock对象的行为依赖风险.在对真实对象进行重构的时候,容易带来该问题.
    • Mock对象的setup过程可能过于繁重.
      • 另一个角度,过于复杂的Mock对象的setup过程,说明真实对象承担了过多的职责.
      • 分出更多职责清晰的小类,可以避免这种情况.
    • Mock对象的setup过程含有过多的语义.
      • Mock对象的行为定义.调用方法的返回值;调用方法时的Throw Exception;给调用方法时传递的参数发送消息.
      • Test Case期望assert的内容.方法是否被调用以及被调用的次数.调用方法时的参数是否合法.

 

posted @ 2014-06-18 10:58  robynhan  阅读(715)  评论(0编辑  收藏  举报