mock打桩之EasyMock

    TDD是测试驱动开发(Test-Driven Development)的英文简称,是敏捷开发中的一项核心实践和技术,也是一种设计方法论。TDD的原理是在开发功能代码之前,先编写单元测试用例代码,测试代码确定需要编写什么产品代码。TDD虽是敏捷方法的核心实践,但不只使用于XP(Extreme Programming),同样可以适用于其他开发方法和过程。

    打桩(mock)是单元测试的重要内容。说难点,谈不上吧,能说出来想到的都不算是难点了。

    mock有两种。一种是静态打桩,一种是动态打桩。

    静态打桩就是在写测试代码之前根据需要打桩的类生成另外一个类,这个类就是mock object。

     动态打桩就是mock object是在测试代码运行的时候才生成的。比较常用的mock工具有EasyMock、Jmock、PowerMock、MockMvc。

    先说EasyMock,话说十年前,有天我同事跟我说他搜索easymock,发现百度排名第一的文章是我写的blog。好吧,侧面证明了当时关于这方面的资料是匮乏的。

easymock使用

    依赖

    代码中mock:

EasyMock设计

  • Mock对象能够模拟其他协同模块的行为,被测试模块通过与Mock对象协作,可以获得一个孤立的测试环境。
  • 使用Mock对象可以模拟在应用用不容易构造(如HttpServletRequest必须在Servlet容器中才能构造出来)和比较复杂的对象(如JDBC中的ResultSet对象),从而使测试顺利进行。
  • Mock对象可以根据享有的接口或类动态生成,不仅能避免额外的编码工作,同时也降低了引入错误的可能。
  • 简单即正义
  • 提供对接口的模拟,能够通过录制、回放、检查三步来完成大体的测试过程。

EasyMock源码解析

Mock对象的创建有两种方式:一种是通过EasyMock类提供的createMock方法创建,另一种是通过EasyMock类的createControl方法得到一个IMocksControl实例,再由这个IMocksControl实例创建Mock实例。

    而实际上这两种方法内部实现是一样的。

    MocksControl类中包含了两个重要的成员变量,分别是接口IMockBehavior和IMocksControlState的实例。

    IMocksBehavior的实现类MocksBehavior是EasyMock的核心类,它保存着一个ExpectedInvocationAndResult对象的一个列表,而ExpectedInvocationAndResult对象中包含着Mock对象方法调用和预期结果的映射。MocksBehavior类提供了addExpected和addActual方法用于添加预期行为和实际调用。

    MocksControl类中包含的另一个成员变量是IMocksControlState实例。IMocksControlState拥有两个不同的实现类:RecordState和ReplayState。顾名思义,RecordState是Mock对象在Record状态时的支持类,它提供了invoke方法在Record状态下的实现。他还提供了andReturn、andThrow、times等方法的实现。在ReplayState中,andReturn、andThrow、times等方法的实现都是抛出IllegalStateException,因为在Replay阶段,开发人员不应该再调用这些方法。

    当调用MocksControl的createMock方法时,该方法首选会生成一个JavaProxyFactory类的对象。JavaProxyFactory是接口IProxyFactory的实现类,它的主要功能就是通过java.lang.reflect.Proxy对指定的接口创建动态代理实例,也就是开发人员在外部看到的Mock对象。

    在创建动态代码的同时,应当提供InvocationHandler的实现类。MockInvocationHandler实现了这个接口,它的invoke方法主要的功能是根据Mock对象状态的不同而分别调用RecordState的invoke实现或是ReplayState的invoke实现。

    下面是备受吐槽的手绘时间:

简单再解释一下这张图:

    当EasyMock类的createMock方法被调用时,它首先创建一个MocksControl对象,并调用该对象的createMock方法创建一个JavaProxyFactory对象和一个MockInvocationHandler对象。JavaProxyFactory对象将MockInvocationHandler作为参数,通过java.lang.reflect.Proxy类的newProxyInstance静态方法创建一个动态代理。

    上面介绍的EasyMock创建的源码解析。可以参考上面的思路再看一下记录Mock对象预期行为的源码,在Replay状态下调用Mock对象的源码。

总结

    EasyMock是一个使用简单,源码也非常简单的工具。如果看spring系列源码有困难的同学可以看一下EasyMock的源码,思考一下EasyMock的设计理念和设计模式。

招贤纳士:

  • 新美大基础架构部招聘高级开发/专家。详细信息请点击公众号内“招贤纳士”菜单。
  • 爱奇艺招聘后台开发工程师,有意向请将简历发到BOSS邮箱:275824717@qq.com

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Java技术栈

你必须掌握的 21 个 Java 核心技术!

写这篇文章的目的是想总结一下自己这么多年来使用java的一些心得体会,希望可以给大家一些经验,能让大家更好学习和使用Java。

1071
来自专栏java一日一条

优秀的Java程序测试是什么样的?

测试的名字至关重要,特别是从文档角度来看的话。我们应该能够大声读出测试的名字就像一组需求一样。事实上,有一个伟大的IntelliJ插件,叫Enso,它会将你的测...

602
来自专栏一枝花算不算浪漫

Java中 VO、 PO、DO、DTO、 BO、 QO、DAO、POJO的概念

44210
来自专栏玄魂工作室

CTF实战26 CTF题目练习和讲解四

该培训中提及的技术只适用于合法CTF比赛和有合法授权的渗透测试,请勿用于其他非法用途,如用作其他非法用途与本文作者无关

1851
来自专栏JAVA高级架构

Java高级程序员(5年左右)面试的题目集

1 时隔两年 再一次的面临离职找工作,这一次换工作有些许的不舍,也有些许的无奈。个人所在的技术团队不错,两年时间成长了很多,也很不舍这个团队。但是,由于公司...

3886
来自专栏积累沉淀

Java设计模式(二十)----状态模式

状态模式 一、概述 二、结构 三、具体案例 四、认识状态模式 一、概述 定义:允许一个对象的内部状态改变时改变它的行为。对象看起来似...

2135
来自专栏知晓程序

开发 | 让小程序用上「指纹识别」,只需这 3 个接口

最初,小程序文档中只有具体接口调用方法,并没有给出小程序中调用指纹识别的最佳实践。

1180
来自专栏腾讯移动品质中心TMQ的专栏

测试左移之代码评审

最近两年,品质中心极力推动测试工作左移,以期能提前发现产品的问题,降低成本。笔者自认代码基础能力还不错,就想通过代码 Review 来提前发现一些Bug。

1761
来自专栏Java帮帮-微信公众号-技术文章全总结

面试复习大纲(最全面)

Java基础 1.数组中的排序问题(笔试或者机试,前者可能性更大) 2.面向对象的理解 面向对象主要有四个特性: 封装、抽象、继承和多态。 封装:在面向对象语言...

3415
来自专栏CVer

免费资源 | 快来Pick《准备秋招学习笔记》

项目地址:https://github.com/CyC2018/Interview-Notebook

1323

扫码关注云+社区